2010-12-16 16:10:54 +00:00
|
|
|
/*
|
2014-03-07 13:38:51 +00:00
|
|
|
* qemu_hotplug.c: QEMU device hotplug management
|
2010-12-16 16:10:54 +00:00
|
|
|
*
|
2016-04-01 14:40:23 +00:00
|
|
|
* Copyright (C) 2006-2016 Red Hat, Inc.
|
2010-12-16 16:10:54 +00:00
|
|
|
* Copyright (C) 2006 Daniel P. Berrange
|
|
|
|
*
|
|
|
|
* 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 22:30:55 +00:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 10:06:23 +00:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2010-12-16 16:10:54 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include "qemu_hotplug.h"
|
2018-12-13 14:53:50 +00:00
|
|
|
#define LIBVIRT_QEMU_HOTPLUGPRIV_H_ALLOW
|
2013-07-26 10:18:01 +00:00
|
|
|
#include "qemu_hotplugpriv.h"
|
2016-02-16 15:24:35 +00:00
|
|
|
#include "qemu_alias.h"
|
2010-12-16 16:10:54 +00:00
|
|
|
#include "qemu_capabilities.h"
|
|
|
|
#include "qemu_domain.h"
|
2016-02-15 18:08:02 +00:00
|
|
|
#include "qemu_domain_address.h"
|
2010-12-16 16:10:54 +00:00
|
|
|
#include "qemu_command.h"
|
|
|
|
#include "qemu_hostdev.h"
|
2014-09-16 20:50:53 +00:00
|
|
|
#include "qemu_interface.h"
|
2016-04-06 13:57:57 +00:00
|
|
|
#include "qemu_process.h"
|
2016-11-15 15:53:04 +00:00
|
|
|
#include "qemu_security.h"
|
2018-05-16 11:39:22 +00:00
|
|
|
#include "qemu_block.h"
|
Move qemu_audit.h helpers into shared code
The LXC and UML drivers can both make use of auditing. Move
the qemu_audit.{c,h} files to src/conf/domain_audit.{c,h}
* src/conf/domain_audit.c: Rename from src/qemu/qemu_audit.c
* src/conf/domain_audit.h: Rename from src/qemu/qemu_audit.h
* src/Makefile.am: Remove qemu_audit.{c,h}, add domain_audit.{c,h}
* src/qemu/qemu_audit.h, src/qemu/qemu_cgroup.c,
src/qemu/qemu_command.c, src/qemu/qemu_driver.c,
src/qemu/qemu_hotplug.c, src/qemu/qemu_migration.c,
src/qemu/qemu_process.c: Update for changed audit API names
2011-07-04 10:56:13 +00:00
|
|
|
#include "domain_audit.h"
|
2014-11-18 23:55:48 +00:00
|
|
|
#include "netdev_bandwidth_conf.h"
|
2010-12-16 16:10:54 +00:00
|
|
|
#include "domain_nwfilter.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
#include "virlog.h"
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
#include "datatypes.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2012-12-13 14:52:25 +00:00
|
|
|
#include "virpci.h"
|
2011-07-19 18:32:58 +00:00
|
|
|
#include "virfile.h"
|
2013-04-25 16:45:55 +00:00
|
|
|
#include "virprocess.h"
|
2010-12-16 16:10:54 +00:00
|
|
|
#include "qemu_cgroup.h"
|
2010-10-26 14:04:46 +00:00
|
|
|
#include "locking/domain_lock.h"
|
2012-03-28 19:11:09 +00:00
|
|
|
#include "virnetdev.h"
|
|
|
|
#include "virnetdevbridge.h"
|
2012-02-10 21:09:00 +00:00
|
|
|
#include "virnetdevtap.h"
|
2015-03-17 17:46:44 +00:00
|
|
|
#include "virnetdevopenvswitch.h"
|
2015-02-23 20:54:56 +00:00
|
|
|
#include "virnetdevmidonet.h"
|
2012-08-16 15:41:06 +00:00
|
|
|
#include "device_conf.h"
|
2012-12-13 15:25:48 +00:00
|
|
|
#include "virstoragefile.h"
|
2013-04-03 10:36:23 +00:00
|
|
|
#include "virstring.h"
|
2013-07-11 15:11:02 +00:00
|
|
|
#include "virtime.h"
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_QEMU
|
2014-02-28 12:16:17 +00:00
|
|
|
|
|
|
|
VIR_LOG_INIT("qemu.qemu_hotplug");
|
|
|
|
|
2015-06-29 14:19:44 +00:00
|
|
|
#define CHANGE_MEDIA_TIMEOUT 5000
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-07-26 10:18:01 +00:00
|
|
|
/* Wait up to 5 seconds for device removal to finish. */
|
|
|
|
unsigned long long qemuDomainRemoveDeviceWaitTime = 1000ull * 5;
|
|
|
|
|
|
|
|
|
qemu_hotplug: Fix a rare race condition when detaching a device twice
https://bugzilla.redhat.com/show_bug.cgi?id=1623389
If a device is detached twice from the same domain the following
race condition may happen:
1) The first DetachDevice() call will issue "device_del" on qemu
monitor, but since the DEVICE_DELETED event did not arrive in
time, the API ends claiming "Device detach request sent
successfully".
2) The second DetachDevice() therefore still find the device in
the domain and thus proceeds to detaching it again. It calls
EnterMonitor() and qemuMonitorSend() trying to issue "device_del"
command again. This gets both domain lock and monitor lock
released.
3) At this point, qemu sends us the DEVICE_DELETED event which is
going to be handled by the event loop which ends up calling
qemuDomainSignalDeviceRemoval() to determine who is going to
remove the device from domain definition. Whether it is the
caller that marked the device for removal or whether it is going
to be the event processing thread.
4) Because the device was marked for removal,
qemuDomainSignalDeviceRemoval() returns true, which means the
event is to be processed by the thread that has marked the device
for removal (and is currently still trying to issue "device_del"
command)
5) The thread finally issues the "device_del" command, which
fails (obviously) and therefore it calls
qemuDomainResetDeviceRemoval() to reset the device marking and
quits immediately after, NOT removing any device from the domain
definition.
At this point, the device is still present in the domain
definition but doesn't exist in qemu anymore. Worse, there is no
way to remove it from the domain definition.
Solution is to note down that we've seen the event and if the
second "device_del" fails, not take it as a failure but carry on
with the usual execution.
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
ACKed-by: Peter Krempa <pkrempa@redhat.com>
2019-03-14 10:02:52 +00:00
|
|
|
static void
|
|
|
|
qemuDomainResetDeviceRemoval(virDomainObjPtr vm);
|
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
/**
|
|
|
|
* qemuDomainDeleteDevice:
|
|
|
|
* @vm: domain object
|
|
|
|
* @alias: device to remove
|
|
|
|
*
|
|
|
|
* This is a wrapper over qemuMonitorDelDevice() plus enter/exit
|
|
|
|
* monitor calls. This function MUST be used instead of plain
|
|
|
|
* qemuMonitorDelDevice() in all places where @alias represents a
|
|
|
|
* device from domain XML, i.e. caller marks the device for
|
|
|
|
* removal and then calls qemuDomainWaitForDeviceRemoval()
|
|
|
|
* followed by qemuDomainRemove*Device().
|
|
|
|
*
|
|
|
|
* For collateral devices (e.g. extension devices like zPCI) it
|
|
|
|
* is safe to use plain qemuMonitorDelDevice().
|
|
|
|
*
|
|
|
|
* Upon entry, @vm must be locked.
|
|
|
|
*
|
|
|
|
* Returns: 0 on success,
|
|
|
|
* -1 otherwise.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
qemuDomainDeleteDevice(virDomainObjPtr vm,
|
|
|
|
const char *alias)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virQEMUDriverPtr driver = priv->driver;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
|
|
|
rc = qemuMonitorDelDevice(priv->mon, alias);
|
|
|
|
|
qemu_hotplug: Fix a rare race condition when detaching a device twice
https://bugzilla.redhat.com/show_bug.cgi?id=1623389
If a device is detached twice from the same domain the following
race condition may happen:
1) The first DetachDevice() call will issue "device_del" on qemu
monitor, but since the DEVICE_DELETED event did not arrive in
time, the API ends claiming "Device detach request sent
successfully".
2) The second DetachDevice() therefore still find the device in
the domain and thus proceeds to detaching it again. It calls
EnterMonitor() and qemuMonitorSend() trying to issue "device_del"
command again. This gets both domain lock and monitor lock
released.
3) At this point, qemu sends us the DEVICE_DELETED event which is
going to be handled by the event loop which ends up calling
qemuDomainSignalDeviceRemoval() to determine who is going to
remove the device from domain definition. Whether it is the
caller that marked the device for removal or whether it is going
to be the event processing thread.
4) Because the device was marked for removal,
qemuDomainSignalDeviceRemoval() returns true, which means the
event is to be processed by the thread that has marked the device
for removal (and is currently still trying to issue "device_del"
command)
5) The thread finally issues the "device_del" command, which
fails (obviously) and therefore it calls
qemuDomainResetDeviceRemoval() to reset the device marking and
quits immediately after, NOT removing any device from the domain
definition.
At this point, the device is still present in the domain
definition but doesn't exist in qemu anymore. Worse, there is no
way to remove it from the domain definition.
Solution is to note down that we've seen the event and if the
second "device_del" fails, not take it as a failure but carry on
with the usual execution.
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
ACKed-by: Peter Krempa <pkrempa@redhat.com>
2019-03-14 10:02:52 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
/* Domain is no longer running. No cleanup needed. */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc < 0) {
|
|
|
|
/* Deleting device failed. Let's check if DEVICE_DELETED
|
|
|
|
* even arrived. If it did, we need to claim success to
|
|
|
|
* make the caller remove device from domain XML. */
|
|
|
|
|
|
|
|
if (priv->unplug.eventSeen) {
|
|
|
|
/* The event arrived. Return success. */
|
|
|
|
VIR_DEBUG("Detaching of device %s failed, but event arrived", alias);
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
|
|
|
rc = 0;
|
|
|
|
} else if (rc == -2) {
|
|
|
|
/* The device does not exist in qemu, but it still
|
|
|
|
* exists in libvirt. Claim success to make caller
|
|
|
|
* qemuDomainWaitForDeviceRemoval(). Otherwise if
|
|
|
|
* domain XML is queried right after detach API the
|
|
|
|
* device would still be there. */
|
|
|
|
VIR_DEBUG("Detaching of device %s failed and no event arrived", alias);
|
|
|
|
rc = 0;
|
|
|
|
}
|
|
|
|
}
|
2019-03-12 12:49:24 +00:00
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-05 12:53:01 +00:00
|
|
|
/**
|
2019-01-23 13:28:31 +00:00
|
|
|
* qemuHotplugPrepareDiskSourceAccess:
|
2014-08-05 12:53:01 +00:00
|
|
|
* @driver: qemu driver struct
|
|
|
|
* @vm: domain object
|
2019-01-23 13:28:31 +00:00
|
|
|
* @src: Source to prepare
|
|
|
|
* @teardown: Teardown the access to @src instead of adding it to a vm
|
2014-08-05 12:53:01 +00:00
|
|
|
*
|
2019-01-23 13:28:31 +00:00
|
|
|
* Setup the locks, cgroups and security permissions on a disk source and its
|
|
|
|
* backing chain. If @teardown is true, then the labels and cgroups are removed
|
|
|
|
* instead.
|
2014-08-05 12:53:01 +00:00
|
|
|
*
|
|
|
|
* Returns 0 on success and -1 on error. Reports libvirt error.
|
|
|
|
*/
|
|
|
|
static int
|
2019-01-23 13:28:31 +00:00
|
|
|
qemuHotplugPrepareDiskSourceAccess(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virStorageSourcePtr src,
|
|
|
|
bool teardown)
|
2014-08-05 12:53:01 +00:00
|
|
|
{
|
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2019-01-23 13:28:31 +00:00
|
|
|
const char *srcstr = NULLSTR(src->path);
|
2014-08-05 12:53:01 +00:00
|
|
|
int ret = -1;
|
2017-09-11 13:28:15 +00:00
|
|
|
virErrorPtr orig_err = NULL;
|
2014-08-05 12:53:01 +00:00
|
|
|
|
|
|
|
/* just tear down the disk access */
|
|
|
|
if (teardown) {
|
2017-09-11 13:28:15 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2014-08-05 12:53:01 +00:00
|
|
|
ret = 0;
|
|
|
|
goto rollback_cgroup;
|
|
|
|
}
|
|
|
|
|
2019-01-23 13:28:31 +00:00
|
|
|
if (virDomainLockImageAttach(driver->lockManager, cfg->uri, vm, src) < 0)
|
2014-08-05 12:53:01 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2019-01-23 13:28:31 +00:00
|
|
|
if (qemuDomainNamespaceSetupDisk(vm, src) < 0)
|
2014-08-05 12:53:01 +00:00
|
|
|
goto rollback_lock;
|
|
|
|
|
2019-01-23 13:28:31 +00:00
|
|
|
if (qemuSecuritySetImageLabel(driver, vm, src, true) < 0)
|
2017-01-19 16:04:15 +00:00
|
|
|
goto rollback_namespace;
|
2014-08-05 12:53:01 +00:00
|
|
|
|
2019-01-23 13:28:31 +00:00
|
|
|
if (qemuSetupImageChainCgroup(vm, src) < 0)
|
2017-01-19 16:04:15 +00:00
|
|
|
goto rollback_label;
|
2016-11-15 15:53:04 +00:00
|
|
|
|
2014-08-05 12:53:01 +00:00
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
rollback_cgroup:
|
2019-01-23 13:28:31 +00:00
|
|
|
if (qemuTeardownImageChainCgroup(vm, src) < 0)
|
|
|
|
VIR_WARN("Unable to tear down cgroup access on %s", srcstr);
|
2014-08-05 12:53:01 +00:00
|
|
|
rollback_label:
|
2019-01-23 13:28:31 +00:00
|
|
|
if (qemuSecurityRestoreImageLabel(driver, vm, src, true) < 0)
|
|
|
|
VIR_WARN("Unable to restore security label on %s", srcstr);
|
2014-08-05 12:53:01 +00:00
|
|
|
|
2017-01-19 16:04:15 +00:00
|
|
|
rollback_namespace:
|
2019-01-23 13:28:31 +00:00
|
|
|
if (qemuDomainNamespaceTeardownDisk(vm, src) < 0)
|
|
|
|
VIR_WARN("Unable to remove /dev entry for %s", srcstr);
|
2017-01-19 16:04:15 +00:00
|
|
|
|
2014-08-05 12:53:01 +00:00
|
|
|
rollback_lock:
|
2019-01-23 13:28:31 +00:00
|
|
|
if (virDomainLockImageDetach(driver->lockManager, vm, src) < 0)
|
|
|
|
VIR_WARN("Unable to release lock on %s", srcstr);
|
2014-08-05 12:53:01 +00:00
|
|
|
|
|
|
|
cleanup:
|
2017-09-11 13:28:15 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2014-08-05 12:53:01 +00:00
|
|
|
virObjectUnref(cfg);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-08 11:00:30 +00:00
|
|
|
static int
|
|
|
|
qemuDomainAttachZPCIDevice(qemuMonitorPtr mon,
|
|
|
|
virDomainDeviceInfoPtr info)
|
|
|
|
{
|
|
|
|
char *devstr_zpci = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(devstr_zpci = qemuBuildZPCIDevStr(info)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorAddDevice(mon, devstr_zpci) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(devstr_zpci);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuDomainDetachZPCIDevice(qemuMonitorPtr mon,
|
|
|
|
virDomainDeviceInfoPtr info)
|
|
|
|
{
|
|
|
|
char *zpciAlias = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (virAsprintf(&zpciAlias, "zpci%d", info->addr.pci.zpci.uid) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuMonitorDelDevice(mon, zpciAlias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(zpciAlias);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuDomainAttachExtensionDevice(qemuMonitorPtr mon,
|
|
|
|
virDomainDeviceInfoPtr info)
|
|
|
|
{
|
|
|
|
if (info->type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI ||
|
|
|
|
info->addr.pci.extFlags == VIR_PCI_ADDRESS_EXTENSION_NONE) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info->addr.pci.extFlags & VIR_PCI_ADDRESS_EXTENSION_ZPCI)
|
|
|
|
return qemuDomainAttachZPCIDevice(mon, info);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuDomainDetachExtensionDevice(qemuMonitorPtr mon,
|
|
|
|
virDomainDeviceInfoPtr info)
|
|
|
|
{
|
|
|
|
if (info->type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI ||
|
|
|
|
info->addr.pci.extFlags == VIR_PCI_ADDRESS_EXTENSION_NONE) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info->addr.pci.extFlags & VIR_PCI_ADDRESS_EXTENSION_ZPCI)
|
|
|
|
return qemuDomainDetachZPCIDevice(mon, info);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-05-23 12:50:17 +00:00
|
|
|
static int
|
2018-07-13 15:55:59 +00:00
|
|
|
qemuHotplugWaitForTrayEject(virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk)
|
2016-05-23 12:50:17 +00:00
|
|
|
{
|
|
|
|
unsigned long long now;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
if (virTimeMillisNow(&now) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
while (disk->tray_status != VIR_DOMAIN_DISK_TRAY_OPEN) {
|
|
|
|
if ((rc = virDomainObjWaitUntil(vm, now + CHANGE_MEDIA_TIMEOUT)) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (rc > 0) {
|
2016-05-23 14:32:06 +00:00
|
|
|
/* the caller called qemuMonitorEjectMedia which usually reports an
|
|
|
|
* error. Report the failure in an off-chance that it didn't. */
|
2018-05-05 12:04:21 +00:00
|
|
|
if (virGetLastErrorCode() == VIR_ERR_OK) {
|
2018-07-13 15:55:59 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("timed out waiting to open tray of '%s'"),
|
|
|
|
disk->dst);
|
2016-05-23 14:32:06 +00:00
|
|
|
}
|
2016-05-23 12:50:17 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-08-08 08:55:30 +00:00
|
|
|
/**
|
2018-07-13 15:44:32 +00:00
|
|
|
* qemuDomainChangeMediaLegacy:
|
2014-08-08 08:55:30 +00:00
|
|
|
* @driver: qemu driver structure
|
|
|
|
* @vm: domain definition
|
|
|
|
* @disk: disk definition to change the source of
|
|
|
|
* @newsrc: new disk source to change to
|
|
|
|
* @force: force the change of media
|
|
|
|
*
|
|
|
|
* Change the media in an ejectable device to the one described by
|
|
|
|
* @newsrc. This function also removes the old source from the
|
|
|
|
* shared device table if appropriate. Note that newsrc is consumed
|
|
|
|
* on success and the old source is freed on success.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on error and reports libvirt error
|
|
|
|
*/
|
2018-07-13 15:44:32 +00:00
|
|
|
static int
|
|
|
|
qemuDomainChangeMediaLegacy(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk,
|
|
|
|
virStorageSourcePtr newsrc,
|
|
|
|
bool force)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
2015-06-29 14:19:44 +00:00
|
|
|
int ret = -1, rc;
|
2010-12-16 16:10:54 +00:00
|
|
|
char *driveAlias = NULL;
|
2011-05-04 12:09:09 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-05-19 13:30:12 +00:00
|
|
|
qemuDomainDiskPrivatePtr diskPriv = QEMU_DOMAIN_DISK_PRIVATE(disk);
|
2014-08-05 11:43:57 +00:00
|
|
|
const char *format = NULL;
|
2014-08-08 08:16:32 +00:00
|
|
|
char *sourcestr = NULL;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2014-08-05 11:43:57 +00:00
|
|
|
if (!disk->info.alias) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2014-08-05 11:43:57 +00:00
|
|
|
_("missing disk device alias name for %s"), disk->dst);
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2018-05-31 09:55:24 +00:00
|
|
|
if (!(driveAlias = qemuAliasDiskDriveFromDisk(disk)))
|
2018-07-13 15:44:32 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2016-05-23 12:50:17 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuMonitorEjectMedia(priv->mon, driveAlias, force);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
2014-05-19 13:48:43 +00:00
|
|
|
|
2019-02-07 11:17:51 +00:00
|
|
|
/* If the tray is present wait for it to open. */
|
|
|
|
if (!force && diskPriv->tray) {
|
2018-07-13 15:55:59 +00:00
|
|
|
rc = qemuHotplugWaitForTrayEject(vm, disk);
|
2016-07-08 10:30:26 +00:00
|
|
|
if (rc < 0)
|
2018-07-13 15:44:32 +00:00
|
|
|
goto cleanup;
|
2018-07-13 15:55:59 +00:00
|
|
|
|
|
|
|
/* re-issue ejection command to pop out the media */
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuMonitorEjectMedia(priv->mon, driveAlias, false);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0 || rc < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-05-23 12:50:17 +00:00
|
|
|
} else {
|
|
|
|
/* otherwise report possible errors from the attempt to eject the media*/
|
|
|
|
if (rc < 0)
|
2018-07-13 15:44:32 +00:00
|
|
|
goto cleanup;
|
2016-05-23 12:50:17 +00:00
|
|
|
}
|
2013-05-28 14:23:00 +00:00
|
|
|
|
2015-03-12 15:57:56 +00:00
|
|
|
if (!virStorageSourceIsEmpty(newsrc)) {
|
2018-10-04 12:34:01 +00:00
|
|
|
if (qemuGetDriveSourceString(newsrc, NULL, &sourcestr) < 0)
|
2018-07-13 15:44:32 +00:00
|
|
|
goto cleanup;
|
2014-08-08 08:16:32 +00:00
|
|
|
|
2018-10-04 13:32:37 +00:00
|
|
|
if (virStorageSourceGetActualType(newsrc) != VIR_STORAGE_TYPE_DIR)
|
|
|
|
format = virStorageFileFormatTypeToString(newsrc->format);
|
|
|
|
|
2013-05-20 17:26:14 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2015-06-29 14:19:44 +00:00
|
|
|
rc = qemuMonitorChangeMedia(priv->mon,
|
|
|
|
driveAlias,
|
|
|
|
sourcestr,
|
|
|
|
format);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
2014-12-16 09:40:58 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
2014-08-05 12:09:44 +00:00
|
|
|
|
2015-06-29 14:19:44 +00:00
|
|
|
if (rc < 0)
|
2018-07-13 15:44:32 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2015-06-29 14:19:44 +00:00
|
|
|
ret = 0;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2013-01-10 21:03:14 +00:00
|
|
|
VIR_FREE(driveAlias);
|
2014-08-08 08:16:32 +00:00
|
|
|
VIR_FREE(sourcestr);
|
2010-12-16 16:10:54 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-09-13 13:49:50 +00:00
|
|
|
|
2018-04-23 11:21:03 +00:00
|
|
|
/**
|
2018-07-13 12:14:17 +00:00
|
|
|
* qemuHotplugAttachManagedPR:
|
|
|
|
* @driver: QEMU driver object
|
2018-04-23 11:21:03 +00:00
|
|
|
* @vm: domain object
|
2018-07-13 12:14:17 +00:00
|
|
|
* @src: new disk source to be attached to @vm
|
|
|
|
* @asyncJob: asynchronous job identifier
|
2018-04-23 11:21:03 +00:00
|
|
|
*
|
2018-05-31 11:56:35 +00:00
|
|
|
* Checks if it's needed to start qemu-pr-helper and add the corresponding
|
|
|
|
* pr-manager-helper object.
|
2018-04-23 11:21:03 +00:00
|
|
|
*
|
2018-07-13 12:14:17 +00:00
|
|
|
* Returns: 0 on success, -1 on error.
|
2018-04-23 11:21:03 +00:00
|
|
|
*/
|
|
|
|
static int
|
2018-07-13 12:14:17 +00:00
|
|
|
qemuHotplugAttachManagedPR(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virStorageSourcePtr src,
|
|
|
|
qemuDomainAsyncJob asyncJob)
|
2018-04-23 11:21:03 +00:00
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2018-05-31 11:56:35 +00:00
|
|
|
virJSONValuePtr props = NULL;
|
2018-07-13 12:14:17 +00:00
|
|
|
bool daemonStarted = false;
|
2018-05-31 11:56:35 +00:00
|
|
|
int ret = -1;
|
2018-07-13 12:14:17 +00:00
|
|
|
int rc;
|
2018-04-23 11:21:03 +00:00
|
|
|
|
2018-05-31 11:56:35 +00:00
|
|
|
if (priv->prDaemonRunning ||
|
2018-07-13 12:14:17 +00:00
|
|
|
!virStorageSourceChainHasManagedPR(src))
|
2018-04-23 11:21:03 +00:00
|
|
|
return 0;
|
|
|
|
|
2018-05-31 11:56:35 +00:00
|
|
|
if (!(props = qemuBuildPRManagedManagerInfoProps(priv)))
|
2018-04-23 11:21:03 +00:00
|
|
|
return -1;
|
|
|
|
|
2018-05-31 11:56:35 +00:00
|
|
|
if (qemuProcessStartManagedPRDaemon(vm) < 0)
|
|
|
|
goto cleanup;
|
2018-04-23 11:21:03 +00:00
|
|
|
|
2018-07-13 12:14:17 +00:00
|
|
|
daemonStarted = true;
|
|
|
|
|
|
|
|
if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
rc = qemuMonitorAddObject(priv->mon, &props, NULL);
|
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0 || rc < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-31 11:56:35 +00:00
|
|
|
ret = 0;
|
2018-05-31 11:29:45 +00:00
|
|
|
|
2018-05-31 11:56:35 +00:00
|
|
|
cleanup:
|
2018-07-13 12:14:17 +00:00
|
|
|
if (ret < 0 && daemonStarted)
|
|
|
|
qemuProcessKillManagedPRDaemon(vm);
|
2018-05-31 11:56:35 +00:00
|
|
|
virJSONValueFree(props);
|
|
|
|
return ret;
|
2018-04-23 11:21:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-11 12:24:49 +00:00
|
|
|
/**
|
|
|
|
* qemuHotplugRemoveManagedPR:
|
|
|
|
* @driver: QEMU driver object
|
|
|
|
* @vm: domain object
|
|
|
|
* @asyncJob: asynchronous job identifier
|
|
|
|
*
|
|
|
|
* Removes the managed PR object from @vm if the configuration does not require
|
|
|
|
* it any more.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
qemuHotplugRemoveManagedPR(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
qemuDomainAsyncJob asyncJob)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virErrorPtr orig_err;
|
2018-07-23 13:38:02 +00:00
|
|
|
int ret = -1;
|
2018-07-11 12:24:49 +00:00
|
|
|
|
|
|
|
if (!priv->prDaemonRunning ||
|
|
|
|
virDomainDefHasManagedPR(vm->def))
|
|
|
|
return 0;
|
|
|
|
|
2018-07-23 13:38:02 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
|
|
|
|
2018-07-11 12:24:49 +00:00
|
|
|
if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0)
|
2018-07-23 13:38:02 +00:00
|
|
|
goto cleanup;
|
2018-07-11 12:24:49 +00:00
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, qemuDomainGetManagedPRAlias()));
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
2018-07-23 13:38:02 +00:00
|
|
|
goto cleanup;
|
2018-07-11 12:24:49 +00:00
|
|
|
|
|
|
|
qemuProcessKillManagedPRDaemon(vm);
|
|
|
|
|
2018-07-23 13:38:02 +00:00
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
virErrorRestore(&orig_err);
|
|
|
|
return ret;
|
2018-07-11 12:24:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-11 16:26:12 +00:00
|
|
|
struct _qemuHotplugDiskSourceData {
|
|
|
|
qemuBlockStorageSourceAttachDataPtr *backends;
|
|
|
|
size_t nbackends;
|
2018-07-13 12:48:59 +00:00
|
|
|
|
|
|
|
/* disk copy-on-read object */
|
|
|
|
virJSONValuePtr corProps;
|
|
|
|
char *corAlias;
|
2018-07-11 16:26:12 +00:00
|
|
|
};
|
|
|
|
typedef struct _qemuHotplugDiskSourceData qemuHotplugDiskSourceData;
|
|
|
|
typedef qemuHotplugDiskSourceData *qemuHotplugDiskSourceDataPtr;
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemuHotplugDiskSourceDataFree(qemuHotplugDiskSourceDataPtr data)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!data)
|
|
|
|
return;
|
|
|
|
|
2018-07-13 12:48:59 +00:00
|
|
|
virJSONValueFree(data->corProps);
|
|
|
|
VIR_FREE(data->corAlias);
|
|
|
|
|
2018-07-11 16:26:12 +00:00
|
|
|
for (i = 0; i < data->nbackends; i++)
|
|
|
|
qemuBlockStorageSourceAttachDataFree(data->backends[i]);
|
|
|
|
|
|
|
|
VIR_FREE(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-11 09:16:41 +00:00
|
|
|
/**
|
|
|
|
* qemuDomainRemoveDiskStorageSourcePrepareData:
|
|
|
|
* @src: disk source structure
|
|
|
|
* @driveAlias: Alias of the -drive backend, the pointer is always consumed
|
|
|
|
*
|
|
|
|
* Prepare qemuBlockStorageSourceAttachDataPtr for detaching a single source
|
|
|
|
* from a VM. If @driveAlias is NULL -blockdev is assumed.
|
|
|
|
*/
|
|
|
|
static qemuBlockStorageSourceAttachDataPtr
|
|
|
|
qemuHotplugRemoveStorageSourcePrepareData(virStorageSourcePtr src,
|
|
|
|
char *driveAlias)
|
|
|
|
|
|
|
|
{
|
|
|
|
qemuDomainStorageSourcePrivatePtr srcpriv = QEMU_DOMAIN_STORAGE_SOURCE_PRIVATE(src);
|
|
|
|
qemuBlockStorageSourceAttachDataPtr data;
|
|
|
|
qemuBlockStorageSourceAttachDataPtr ret = NULL;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(data) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (driveAlias) {
|
|
|
|
VIR_STEAL_PTR(data->driveAlias, driveAlias);
|
|
|
|
data->driveAdded = true;
|
|
|
|
} else {
|
|
|
|
data->formatNodeName = src->nodeformat;
|
|
|
|
data->formatAttached = true;
|
|
|
|
data->storageNodeName = src->nodestorage;
|
|
|
|
data->storageAttached = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (src->pr &&
|
|
|
|
!virStoragePRDefIsManaged(src->pr) &&
|
|
|
|
VIR_STRDUP(data->prmgrAlias, src->pr->mgralias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(data->tlsAlias, src->tlsAlias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (srcpriv) {
|
|
|
|
if (srcpriv->secinfo &&
|
|
|
|
srcpriv->secinfo->type == VIR_DOMAIN_SECRET_INFO_TYPE_AES &&
|
|
|
|
VIR_STRDUP(data->authsecretAlias, srcpriv->secinfo->s.aes.alias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (srcpriv->encinfo &&
|
|
|
|
srcpriv->encinfo->type == VIR_DOMAIN_SECRET_INFO_TYPE_AES &&
|
|
|
|
VIR_STRDUP(data->encryptsecretAlias, srcpriv->encinfo->s.aes.alias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_STEAL_PTR(ret, data);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(driveAlias);
|
|
|
|
qemuBlockStorageSourceAttachDataFree(data);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static qemuHotplugDiskSourceDataPtr
|
|
|
|
qemuHotplugDiskSourceRemovePrepare(virDomainDiskDefPtr disk,
|
2018-10-04 15:49:00 +00:00
|
|
|
virStorageSourcePtr src,
|
2018-07-13 12:48:59 +00:00
|
|
|
virQEMUCapsPtr qemuCaps)
|
2018-07-11 09:16:41 +00:00
|
|
|
{
|
2018-07-13 12:48:59 +00:00
|
|
|
qemuDomainDiskPrivatePtr diskPriv = QEMU_DOMAIN_DISK_PRIVATE(disk);
|
2018-07-11 09:16:41 +00:00
|
|
|
qemuBlockStorageSourceAttachDataPtr backend = NULL;
|
|
|
|
qemuHotplugDiskSourceDataPtr data = NULL;
|
|
|
|
qemuHotplugDiskSourceDataPtr ret = NULL;
|
|
|
|
char *drivealias = NULL;
|
2018-07-13 12:48:59 +00:00
|
|
|
virStorageSourcePtr n;
|
2018-07-11 09:16:41 +00:00
|
|
|
|
|
|
|
if (VIR_ALLOC(data) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2018-07-13 12:48:59 +00:00
|
|
|
if (virQEMUCapsGet(qemuCaps, QEMU_CAPS_BLOCKDEV)) {
|
|
|
|
if (VIR_STRDUP(data->corAlias, diskPriv->nodeCopyOnRead) < 0)
|
|
|
|
goto cleanup;
|
2018-07-11 09:16:41 +00:00
|
|
|
|
2018-10-04 15:49:00 +00:00
|
|
|
for (n = src; virStorageSourceIsBacking(n); n = n->backingStore) {
|
2018-07-13 12:48:59 +00:00
|
|
|
if (!(backend = qemuHotplugRemoveStorageSourcePrepareData(n, NULL)))
|
|
|
|
goto cleanup;
|
2018-07-11 09:16:41 +00:00
|
|
|
|
2018-07-13 12:48:59 +00:00
|
|
|
if (VIR_APPEND_ELEMENT(data->backends, data->nbackends, backend) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!(drivealias = qemuAliasDiskDriveFromDisk(disk)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-10-04 15:49:00 +00:00
|
|
|
if (!(backend = qemuHotplugRemoveStorageSourcePrepareData(src,
|
2018-07-13 12:48:59 +00:00
|
|
|
drivealias)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (VIR_APPEND_ELEMENT(data->backends, data->nbackends, backend) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-07-11 09:16:41 +00:00
|
|
|
|
|
|
|
VIR_STEAL_PTR(ret, data);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
qemuBlockStorageSourceAttachDataFree(backend);
|
|
|
|
qemuHotplugDiskSourceDataFree(data);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-11 16:26:12 +00:00
|
|
|
/**
|
|
|
|
* qemuHotplugDiskSourceAttachPrepare:
|
|
|
|
* @disk: disk to generate attachment data for
|
2018-10-04 15:49:00 +00:00
|
|
|
* @src: disk source to prepare attachment
|
2018-07-11 16:26:12 +00:00
|
|
|
* @qemuCaps: capabilities of the qemu process
|
|
|
|
*
|
|
|
|
* Prepares and returns qemuHotplugDiskSourceData structure filled with all data
|
|
|
|
* which will fully attach the source backend of the disk to a given VM.
|
|
|
|
*/
|
|
|
|
static qemuHotplugDiskSourceDataPtr
|
|
|
|
qemuHotplugDiskSourceAttachPrepare(virDomainDiskDefPtr disk,
|
2018-10-04 15:49:00 +00:00
|
|
|
virStorageSourcePtr src,
|
2018-07-11 16:26:12 +00:00
|
|
|
virQEMUCapsPtr qemuCaps)
|
|
|
|
{
|
2018-07-13 12:48:59 +00:00
|
|
|
qemuBlockStorageSourceAttachDataPtr backend = NULL;
|
2018-07-11 16:26:12 +00:00
|
|
|
qemuHotplugDiskSourceDataPtr data;
|
|
|
|
qemuHotplugDiskSourceDataPtr ret = NULL;
|
2018-10-04 15:49:00 +00:00
|
|
|
virStorageSourcePtr savesrc = NULL;
|
2018-07-13 12:48:59 +00:00
|
|
|
virStorageSourcePtr n;
|
2018-07-11 16:26:12 +00:00
|
|
|
|
|
|
|
if (VIR_ALLOC(data) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2018-07-13 12:48:59 +00:00
|
|
|
if (virQEMUCapsGet(qemuCaps, QEMU_CAPS_BLOCKDEV)) {
|
|
|
|
if (disk->copy_on_read == VIR_TRISTATE_SWITCH_ON &&
|
|
|
|
!(data->corProps = qemuBlockStorageGetCopyOnReadProps(disk)))
|
|
|
|
goto cleanup;
|
2018-07-11 16:26:12 +00:00
|
|
|
|
2018-10-04 15:49:00 +00:00
|
|
|
for (n = src; virStorageSourceIsBacking(n); n = n->backingStore) {
|
2018-07-13 12:48:59 +00:00
|
|
|
if (!(backend = qemuBlockStorageSourceAttachPrepareBlockdev(n)))
|
|
|
|
goto cleanup;
|
2018-07-11 16:26:12 +00:00
|
|
|
|
2018-07-13 12:48:59 +00:00
|
|
|
if (qemuBuildStorageSourceAttachPrepareCommon(n, backend, qemuCaps) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (VIR_APPEND_ELEMENT(data->backends, data->nbackends, backend) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
} else {
|
2018-10-04 15:49:00 +00:00
|
|
|
VIR_STEAL_PTR(savesrc, disk->src);
|
|
|
|
disk->src = src;
|
|
|
|
|
2018-07-13 12:48:59 +00:00
|
|
|
if (!(backend = qemuBuildStorageSourceAttachPrepareDrive(disk, qemuCaps)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-10-04 15:49:00 +00:00
|
|
|
VIR_STEAL_PTR(disk->src, savesrc);
|
|
|
|
|
|
|
|
if (qemuBuildStorageSourceAttachPrepareCommon(src, backend, qemuCaps) < 0)
|
2018-07-13 12:48:59 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (VIR_APPEND_ELEMENT(data->backends, data->nbackends, backend) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-07-11 16:26:12 +00:00
|
|
|
|
|
|
|
VIR_STEAL_PTR(ret, data);
|
|
|
|
|
|
|
|
cleanup:
|
2018-10-04 15:49:00 +00:00
|
|
|
if (savesrc)
|
|
|
|
VIR_STEAL_PTR(disk->src, savesrc);
|
|
|
|
|
2018-07-11 16:26:12 +00:00
|
|
|
qemuBlockStorageSourceAttachDataFree(backend);
|
|
|
|
qemuHotplugDiskSourceDataFree(data);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qemuHotplugDiskSourceAttach:
|
|
|
|
* @mon: monitor object
|
|
|
|
* @data: disk backend data object describing what to remove
|
|
|
|
*
|
|
|
|
* Attach a disk source backend with all relevant pieces. Caller must enter the
|
|
|
|
* monitor context for @mon.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
qemuHotplugDiskSourceAttach(qemuMonitorPtr mon,
|
|
|
|
qemuHotplugDiskSourceDataPtr data)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = data->nbackends; i > 0; i--) {
|
|
|
|
if (qemuBlockStorageSourceAttachApply(mon, data->backends[i - 1]) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-07-13 12:48:59 +00:00
|
|
|
if (data->corProps &&
|
|
|
|
qemuMonitorAddObject(mon, &data->corProps, &data->corAlias) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2018-07-11 16:26:12 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qemuHotplugDiskSourceRemove:
|
|
|
|
* @mon: monitor object
|
|
|
|
* @data: disk backend data object describing what to remove
|
|
|
|
*
|
|
|
|
* Remove a disk source backend with all relevant pieces. This function
|
|
|
|
* preserves the error which was set prior to calling it. Caller must enter the
|
|
|
|
* monitor context for @mon.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
qemuHotplugDiskSourceRemove(qemuMonitorPtr mon,
|
|
|
|
qemuHotplugDiskSourceDataPtr data)
|
|
|
|
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
2018-07-13 12:48:59 +00:00
|
|
|
if (data->corAlias)
|
|
|
|
ignore_value(qemuMonitorDelObject(mon, data->corAlias));
|
|
|
|
|
2018-07-11 16:26:12 +00:00
|
|
|
for (i = 0; i < data->nbackends; i++)
|
|
|
|
qemuBlockStorageSourceAttachRollback(mon, data->backends[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-13 16:06:09 +00:00
|
|
|
/**
|
|
|
|
* qemuDomainChangeMediaBlockdev:
|
|
|
|
* @driver: qemu driver structure
|
|
|
|
* @vm: domain definition
|
|
|
|
* @disk: disk definition to change the source of
|
2018-10-04 15:56:46 +00:00
|
|
|
* @oldsrc: old source definition
|
2018-07-13 16:06:09 +00:00
|
|
|
* @newsrc: new disk source to change to
|
|
|
|
* @force: force the change of media
|
|
|
|
*
|
|
|
|
* Change the media in an ejectable device to the one described by
|
|
|
|
* @newsrc. This function also removes the old source from the
|
|
|
|
* shared device table if appropriate. Note that newsrc is consumed
|
|
|
|
* on success and the old source is freed on success.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on error and reports libvirt error
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
qemuDomainChangeMediaBlockdev(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk,
|
2018-10-04 15:56:46 +00:00
|
|
|
virStorageSourcePtr oldsrc,
|
2018-07-13 16:06:09 +00:00
|
|
|
virStorageSourcePtr newsrc,
|
|
|
|
bool force)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
qemuDomainDiskPrivatePtr diskPriv = QEMU_DOMAIN_DISK_PRIVATE(disk);
|
|
|
|
qemuHotplugDiskSourceDataPtr newbackend = NULL;
|
|
|
|
qemuHotplugDiskSourceDataPtr oldbackend = NULL;
|
|
|
|
char *nodename = NULL;
|
|
|
|
int rc;
|
|
|
|
int ret = -1;
|
|
|
|
|
2018-10-04 15:56:46 +00:00
|
|
|
if (!virStorageSourceIsEmpty(oldsrc) &&
|
|
|
|
!(oldbackend = qemuHotplugDiskSourceRemovePrepare(disk, oldsrc,
|
2018-10-04 15:49:00 +00:00
|
|
|
priv->qemuCaps)))
|
2018-07-13 16:06:09 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-10-04 15:56:46 +00:00
|
|
|
if (!virStorageSourceIsEmpty(newsrc)) {
|
|
|
|
if (!(newbackend = qemuHotplugDiskSourceAttachPrepare(disk, newsrc,
|
2018-07-13 16:06:09 +00:00
|
|
|
priv->qemuCaps)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuDomainDiskGetBackendAlias(disk, priv->qemuCaps, &nodename) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (diskPriv->tray && disk->tray_status != VIR_DOMAIN_DISK_TRAY_OPEN) {
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuMonitorBlockdevTrayOpen(priv->mon, diskPriv->qomName, force);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0 || rc < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!force && qemuHotplugWaitForTrayEject(vm, disk) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
|
|
|
rc = qemuMonitorBlockdevMediumRemove(priv->mon, diskPriv->qomName);
|
|
|
|
|
|
|
|
if (rc == 0 && oldbackend)
|
|
|
|
qemuHotplugDiskSourceRemove(priv->mon, oldbackend);
|
|
|
|
|
|
|
|
if (newbackend && nodename) {
|
|
|
|
if (rc == 0)
|
|
|
|
rc = qemuHotplugDiskSourceAttach(priv->mon, newbackend);
|
|
|
|
|
|
|
|
if (rc == 0)
|
|
|
|
rc = qemuMonitorBlockdevMediumInsert(priv->mon, diskPriv->qomName,
|
|
|
|
nodename);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rc == 0)
|
|
|
|
rc = qemuMonitorBlockdevTrayClose(priv->mon, diskPriv->qomName);
|
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0 || rc < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
qemuHotplugDiskSourceDataFree(newbackend);
|
|
|
|
qemuHotplugDiskSourceDataFree(oldbackend);
|
2018-08-22 11:09:50 +00:00
|
|
|
VIR_FREE(nodename);
|
2018-07-13 16:06:09 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-13 15:44:32 +00:00
|
|
|
/**
|
|
|
|
* qemuDomainChangeEjectableMedia:
|
|
|
|
* @driver: qemu driver structure
|
|
|
|
* @vm: domain definition
|
|
|
|
* @disk: disk definition to change the source of
|
|
|
|
* @newsrc: new disk source to change to
|
|
|
|
* @force: force the change of media
|
|
|
|
*
|
|
|
|
* Change the media in an ejectable device to the one described by
|
|
|
|
* @newsrc. This function also removes the old source from the
|
|
|
|
* shared device table if appropriate. Note that newsrc is consumed
|
|
|
|
* on success and the old source is freed on success.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on error and reports libvirt error
|
|
|
|
*/
|
2018-09-25 12:21:27 +00:00
|
|
|
int
|
2018-07-13 15:44:32 +00:00
|
|
|
qemuDomainChangeEjectableMedia(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk,
|
|
|
|
virStorageSourcePtr newsrc,
|
|
|
|
bool force)
|
|
|
|
{
|
2018-09-27 14:50:55 +00:00
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2018-07-13 16:06:09 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2018-09-27 14:50:55 +00:00
|
|
|
virStorageSourcePtr oldsrc = disk->src;
|
2018-09-25 12:47:36 +00:00
|
|
|
bool sharedAdded = false;
|
2018-07-13 15:44:32 +00:00
|
|
|
int ret = -1;
|
|
|
|
int rc;
|
|
|
|
|
2018-09-27 14:50:55 +00:00
|
|
|
disk->src = newsrc;
|
|
|
|
|
2018-09-25 12:47:36 +00:00
|
|
|
if (virDomainDiskTranslateSourcePool(disk) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuAddSharedDisk(driver, disk, vm->def->name) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
sharedAdded = true;
|
|
|
|
|
2019-01-16 14:33:07 +00:00
|
|
|
if (qemuDomainDetermineDiskChain(driver, vm, disk, NULL, true) < 0)
|
2018-09-25 12:47:36 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-09-27 14:50:55 +00:00
|
|
|
if (qemuDomainPrepareDiskSource(disk, priv, cfg) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2019-01-23 13:28:31 +00:00
|
|
|
if (qemuHotplugPrepareDiskSourceAccess(driver, vm, newsrc, false) < 0)
|
2018-07-13 15:44:32 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-07-13 16:37:43 +00:00
|
|
|
if (qemuHotplugAttachManagedPR(driver, vm, newsrc, QEMU_ASYNC_JOB_NONE) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-07-13 16:06:09 +00:00
|
|
|
if (virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_BLOCKDEV))
|
2018-09-27 14:50:55 +00:00
|
|
|
rc = qemuDomainChangeMediaBlockdev(driver, vm, disk, oldsrc, newsrc, force);
|
2018-07-13 16:06:09 +00:00
|
|
|
else
|
|
|
|
rc = qemuDomainChangeMediaLegacy(driver, vm, disk, newsrc, force);
|
2018-07-13 15:44:32 +00:00
|
|
|
|
2018-09-27 14:50:55 +00:00
|
|
|
virDomainAuditDisk(vm, oldsrc, newsrc, "update", rc >= 0);
|
2018-07-13 15:44:32 +00:00
|
|
|
|
2018-09-25 12:47:36 +00:00
|
|
|
if (rc < 0)
|
2018-07-13 15:44:32 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* remove the old source from shared device list */
|
2018-09-27 14:50:55 +00:00
|
|
|
disk->src = oldsrc;
|
2018-07-13 15:44:32 +00:00
|
|
|
ignore_value(qemuRemoveSharedDisk(driver, disk, vm->def->name));
|
2019-01-23 13:28:31 +00:00
|
|
|
ignore_value(qemuHotplugPrepareDiskSourceAccess(driver, vm, oldsrc, true));
|
2018-07-13 15:44:32 +00:00
|
|
|
|
2018-09-27 14:50:55 +00:00
|
|
|
/* media was changed, so we can remove the old media definition now */
|
2019-02-15 12:03:58 +00:00
|
|
|
virObjectUnref(oldsrc);
|
2018-09-27 14:50:55 +00:00
|
|
|
oldsrc = NULL;
|
|
|
|
disk->src = newsrc;
|
2018-07-13 16:37:43 +00:00
|
|
|
|
2018-07-13 15:44:32 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
2018-09-25 12:47:36 +00:00
|
|
|
/* undo changes to the new disk */
|
|
|
|
if (ret < 0) {
|
|
|
|
if (sharedAdded)
|
|
|
|
ignore_value(qemuRemoveSharedDisk(driver, disk, vm->def->name));
|
|
|
|
|
2019-01-23 13:28:31 +00:00
|
|
|
ignore_value(qemuHotplugPrepareDiskSourceAccess(driver, vm, newsrc, true));
|
2018-09-25 12:47:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* remove PR manager object if unneeded */
|
|
|
|
ignore_value(qemuHotplugRemoveManagedPR(driver, vm, QEMU_ASYNC_JOB_NONE));
|
|
|
|
|
|
|
|
/* revert old image do the disk definition */
|
2018-09-27 14:50:55 +00:00
|
|
|
if (oldsrc)
|
|
|
|
disk->src = oldsrc;
|
|
|
|
|
|
|
|
virObjectUnref(cfg);
|
2018-07-13 15:44:32 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-10-18 15:17:45 +00:00
|
|
|
/**
|
|
|
|
* qemuDomainAttachDiskGeneric:
|
|
|
|
*
|
|
|
|
* Attaches disk to a VM. This function aggregates common code for all bus types.
|
|
|
|
* In cases when the VM crashed while adding the disk, -2 is returned. */
|
2013-07-18 08:58:01 +00:00
|
|
|
static int
|
2018-02-09 16:14:41 +00:00
|
|
|
qemuDomainAttachDiskGeneric(virQEMUDriverPtr driver,
|
2017-10-18 15:17:45 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
int ret = -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2018-07-11 16:26:12 +00:00
|
|
|
qemuHotplugDiskSourceDataPtr diskdata = NULL;
|
2010-12-16 16:10:54 +00:00
|
|
|
char *devstr = NULL;
|
2018-09-24 14:49:01 +00:00
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2019-01-23 13:28:31 +00:00
|
|
|
if (qemuHotplugPrepareDiskSourceAccess(driver, vm, disk->src, false) < 0)
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2018-06-26 14:55:19 +00:00
|
|
|
if (qemuAssignDeviceDiskAlias(vm->def, disk, priv->qemuCaps) < 0)
|
2016-04-26 11:51:41 +00:00
|
|
|
goto error;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2018-09-24 14:49:01 +00:00
|
|
|
if (qemuDomainPrepareDiskSource(disk, priv, cfg) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2018-10-04 15:49:00 +00:00
|
|
|
if (!(diskdata = qemuHotplugDiskSourceAttachPrepare(disk, disk->src,
|
|
|
|
priv->qemuCaps)))
|
2016-04-26 11:51:41 +00:00
|
|
|
goto error;
|
|
|
|
|
2018-07-10 08:41:04 +00:00
|
|
|
if (!(devstr = qemuBuildDiskDeviceStr(vm->def, disk, 0, priv->qemuCaps)))
|
2016-04-26 11:51:41 +00:00
|
|
|
goto error;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2018-05-10 14:09:20 +00:00
|
|
|
if (VIR_REALLOC_N(vm->def->disks, vm->def->ndisks + 1) < 0)
|
2010-12-16 16:10:54 +00:00
|
|
|
goto error;
|
|
|
|
|
2018-07-13 12:14:17 +00:00
|
|
|
if (qemuHotplugAttachManagedPR(driver, vm, disk->src, QEMU_ASYNC_JOB_NONE) < 0)
|
|
|
|
goto error;
|
2016-04-11 13:16:54 +00:00
|
|
|
|
2018-07-13 12:14:17 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2018-05-31 11:56:35 +00:00
|
|
|
|
2018-07-11 16:26:12 +00:00
|
|
|
if (qemuHotplugDiskSourceAttach(priv->mon, diskdata) < 0)
|
2016-07-14 21:28:53 +00:00
|
|
|
goto exit_monitor;
|
2016-04-11 13:16:54 +00:00
|
|
|
|
2018-11-08 11:00:30 +00:00
|
|
|
if (qemuDomainAttachExtensionDevice(priv->mon, &disk->info) < 0)
|
|
|
|
goto exit_monitor;
|
|
|
|
|
|
|
|
if (qemuMonitorAddDevice(priv->mon, devstr) < 0) {
|
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, &disk->info));
|
2016-07-14 21:28:53 +00:00
|
|
|
goto exit_monitor;
|
2018-11-08 11:00:30 +00:00
|
|
|
}
|
2016-04-11 13:16:54 +00:00
|
|
|
|
2015-01-07 12:12:18 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
2017-10-18 15:17:45 +00:00
|
|
|
ret = -2;
|
2016-07-14 21:28:53 +00:00
|
|
|
goto error;
|
2015-01-07 12:12:18 +00:00
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2016-04-11 13:16:54 +00:00
|
|
|
virDomainAuditDisk(vm, NULL, disk->src, "attach", true);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
virDomainDiskInsertPreAlloced(vm->def, disk);
|
2016-04-11 13:16:54 +00:00
|
|
|
ret = 0;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2018-07-11 16:26:12 +00:00
|
|
|
qemuHotplugDiskSourceDataFree(diskdata);
|
2016-04-06 19:00:59 +00:00
|
|
|
qemuDomainSecretDiskDestroy(disk);
|
2018-04-23 11:21:03 +00:00
|
|
|
VIR_FREE(devstr);
|
2018-09-24 14:49:01 +00:00
|
|
|
virObjectUnref(cfg);
|
2013-01-10 21:03:14 +00:00
|
|
|
return ret;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2016-07-14 21:28:53 +00:00
|
|
|
exit_monitor:
|
2018-07-11 16:26:12 +00:00
|
|
|
qemuHotplugDiskSourceRemove(priv->mon, diskdata);
|
2018-05-16 11:39:22 +00:00
|
|
|
|
2017-02-22 17:39:17 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
2017-10-18 15:17:45 +00:00
|
|
|
ret = -2;
|
2018-07-13 12:14:17 +00:00
|
|
|
if (qemuHotplugRemoveManagedPR(driver, vm, QEMU_ASYNC_JOB_NONE) < 0)
|
|
|
|
ret = -2;
|
2016-04-11 13:16:54 +00:00
|
|
|
|
|
|
|
virDomainAuditDisk(vm, NULL, disk->src, "attach", false);
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
error:
|
2019-01-23 13:28:31 +00:00
|
|
|
ignore_value(qemuHotplugPrepareDiskSourceAccess(driver, vm, disk->src, true));
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-10-18 15:17:45 +00:00
|
|
|
static int
|
2018-02-09 16:14:41 +00:00
|
|
|
qemuDomainAttachVirtioDiskDevice(virQEMUDriverPtr driver,
|
2017-10-18 15:17:45 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk)
|
|
|
|
{
|
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_DISK, { .disk = disk } };
|
|
|
|
bool releaseaddr = false;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
if (qemuDomainEnsureVirtioAddress(&releaseaddr, vm, &dev, disk->dst) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
if ((rv = qemuDomainAttachDiskGeneric(driver, vm, disk)) < 0) {
|
2017-10-18 15:17:45 +00:00
|
|
|
if (rv == -1 && releaseaddr)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &disk->info, disk->dst);
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-21 03:36:25 +00:00
|
|
|
int qemuDomainAttachControllerDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainControllerDefPtr controller)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
const char* type = virDomainControllerTypeToString(controller->type);
|
|
|
|
char *devstr = NULL;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-09-07 16:29:30 +00:00
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_CONTROLLER,
|
|
|
|
{ .controller = controller } };
|
2011-04-26 03:40:01 +00:00
|
|
|
bool releaseaddr = false;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2015-12-09 16:27:12 +00:00
|
|
|
if (controller->type != VIR_DOMAIN_CONTROLLER_TYPE_SCSI) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("'%s' controller cannot be hot plugged."),
|
|
|
|
virDomainControllerTypeToString(controller->type));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-05-10 17:14:32 +00:00
|
|
|
/* default idx would normally be set by virDomainDefPostParse(),
|
|
|
|
* which isn't called in the case of live attach of a single
|
|
|
|
* device.
|
|
|
|
*/
|
|
|
|
if (controller->idx == -1)
|
|
|
|
controller->idx = virDomainControllerFindUnusedIndex(vm->def,
|
|
|
|
controller->type);
|
|
|
|
|
2013-09-24 08:03:15 +00:00
|
|
|
if (virDomainControllerFind(vm->def, controller->type, controller->idx) >= 0) {
|
2012-07-23 08:18:57 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("target %s:%d already exists"),
|
|
|
|
type, controller->idx);
|
|
|
|
return -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2017-10-12 12:53:27 +00:00
|
|
|
if (qemuDomainEnsureVirtioAddress(&releaseaddr, vm, &dev, "controller") < 0)
|
|
|
|
return -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2016-04-26 13:15:03 +00:00
|
|
|
if (qemuAssignDeviceControllerAlias(vm->def, priv->qemuCaps, controller) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2019-01-15 15:31:54 +00:00
|
|
|
if (qemuBuildControllerDevStr(vm->def, controller, priv->qemuCaps, &devstr) < 0)
|
2017-02-28 09:46:30 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!devstr)
|
2016-04-26 13:15:03 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-07-04 10:14:12 +00:00
|
|
|
if (VIR_REALLOC_N(vm->def->controllers, vm->def->ncontrollers+1) < 0)
|
2010-12-16 16:10:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2013-02-06 18:17:20 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2018-11-08 11:00:30 +00:00
|
|
|
|
|
|
|
if ((ret = qemuDomainAttachExtensionDevice(priv->mon,
|
|
|
|
&controller->info)) < 0) {
|
|
|
|
goto exit_monitor;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((ret = qemuMonitorAddDevice(priv->mon, devstr)) < 0)
|
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, &controller->info));
|
|
|
|
|
|
|
|
exit_monitor:
|
2015-01-07 12:12:18 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
releaseaddr = false;
|
|
|
|
ret = -1;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2017-10-12 13:06:56 +00:00
|
|
|
if (ret == 0)
|
2010-12-16 16:10:54 +00:00
|
|
|
virDomainControllerInsertPreAlloced(vm->def, controller);
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2013-07-09 20:30:57 +00:00
|
|
|
if (ret != 0 && releaseaddr)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &controller->info, NULL);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainControllerDefPtr
|
2012-11-28 16:43:10 +00:00
|
|
|
qemuDomainFindOrCreateSCSIDiskController(virQEMUDriverPtr driver,
|
2010-12-16 16:10:54 +00:00
|
|
|
virDomainObjPtr vm,
|
2011-05-04 12:09:09 +00:00
|
|
|
int controller)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
size_t i;
|
2010-12-16 16:10:54 +00:00
|
|
|
virDomainControllerDefPtr cont;
|
2018-01-30 22:29:48 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2017-12-04 19:33:30 +00:00
|
|
|
int model = -1;
|
2011-05-04 12:09:09 +00:00
|
|
|
|
2013-05-21 07:21:20 +00:00
|
|
|
for (i = 0; i < vm->def->ncontrollers; i++) {
|
2010-12-16 16:10:54 +00:00
|
|
|
cont = vm->def->controllers[i];
|
|
|
|
|
|
|
|
if (cont->type != VIR_DOMAIN_CONTROLLER_TYPE_SCSI)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (cont->idx == controller)
|
|
|
|
return cont;
|
2017-12-04 19:33:30 +00:00
|
|
|
|
|
|
|
/* Because virDomainHostdevAssignAddress called during
|
|
|
|
* virDomainHostdevDefPostParse cannot add a new controller
|
|
|
|
* it will assign a controller index to a controller that doesn't
|
|
|
|
* exist leaving this code to perform the magic of adding the
|
|
|
|
* controller. Because that code would be attempting to add a
|
|
|
|
* SCSI disk to an existing controller, let's save the model
|
|
|
|
* of the "last" SCSI controller we find so that if we end up
|
|
|
|
* creating a controller below it uses the same controller model. */
|
|
|
|
model = cont->model;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* No SCSI controller present, for backward compatibility we
|
|
|
|
* now hotplug a controller */
|
2013-07-04 10:14:12 +00:00
|
|
|
if (VIR_ALLOC(cont) < 0)
|
2010-12-16 16:10:54 +00:00
|
|
|
return NULL;
|
|
|
|
cont->type = VIR_DOMAIN_CONTROLLER_TYPE_SCSI;
|
2011-01-31 07:55:40 +00:00
|
|
|
cont->idx = controller;
|
2018-02-14 10:51:26 +00:00
|
|
|
if (model == VIR_DOMAIN_CONTROLLER_MODEL_SCSI_DEFAULT)
|
2018-01-30 22:29:48 +00:00
|
|
|
cont->model = qemuDomainGetSCSIControllerModel(vm->def, cont, priv->qemuCaps);
|
|
|
|
else
|
|
|
|
cont->model = model;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2017-12-04 19:33:30 +00:00
|
|
|
VIR_INFO("No SCSI controller present, hotplugging one model=%s",
|
2018-01-30 22:29:48 +00:00
|
|
|
virDomainControllerModelSCSITypeToString(cont->model));
|
2017-12-04 19:33:30 +00:00
|
|
|
if (qemuDomainAttachControllerDevice(driver, vm, cont) < 0) {
|
2010-12-16 16:10:54 +00:00
|
|
|
VIR_FREE(cont);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!virDomainObjIsActive(vm)) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("guest unexpectedly quit"));
|
2010-12-16 16:10:54 +00:00
|
|
|
/* cont doesn't need freeing here, since the reference
|
|
|
|
* now held in def->controllers */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return cont;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-18 08:58:01 +00:00
|
|
|
static int
|
2018-02-09 16:14:41 +00:00
|
|
|
qemuDomainAttachSCSIDisk(virQEMUDriverPtr driver,
|
2013-07-18 08:58:01 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
2016-06-27 14:43:46 +00:00
|
|
|
size_t i;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
/* We should have an address already, so make sure */
|
|
|
|
if (disk->info.type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_DRIVE) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unexpected disk address type %s"),
|
|
|
|
virDomainDeviceAddressTypeToString(disk->info.type));
|
2017-10-18 15:22:23 +00:00
|
|
|
return -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2016-06-27 14:43:46 +00:00
|
|
|
/* Let's make sure the disk has a controller defined and loaded before
|
|
|
|
* trying to add it. The controller used by the disk must exist before a
|
|
|
|
* qemu command line string is generated.
|
|
|
|
*
|
|
|
|
* Ensure that the given controller and all controllers with a smaller index
|
|
|
|
* exist; there must not be any missing index in between.
|
|
|
|
*/
|
|
|
|
for (i = 0; i <= disk->info.addr.drive.controller; i++) {
|
|
|
|
if (!qemuDomainFindOrCreateSCSIDiskController(driver, vm, i))
|
2017-10-18 15:22:23 +00:00
|
|
|
return -1;
|
2016-07-18 16:50:52 +00:00
|
|
|
}
|
2016-06-02 20:28:28 +00:00
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
if (qemuDomainAttachDiskGeneric(driver, vm, disk) < 0)
|
2017-10-18 15:22:23 +00:00
|
|
|
return -1;
|
qemu: Add TLS support for Veritas HyperScale (VxHS)
Alter qemu command line generation in order to possibly add TLS for
a suitably configured domain.
Sample TLS args generated by libvirt -
-object tls-creds-x509,id=objvirtio-disk0_tls0,dir=/etc/pki/qemu,\
endpoint=client,verify-peer=yes \
-drive file.driver=vxhs,file.tls-creds=objvirtio-disk0_tls0,\
file.vdisk-id=eb90327c-8302-4725-9e1b-4e85ed4dc251,\
file.server.type=tcp,file.server.host=192.168.0.1,\
file.server.port=9999,format=raw,if=none,\
id=drive-virtio-disk0,cache=none \
-device virtio-blk-pci,bus=pci.0,addr=0x4,drive=drive-virtio-disk0,\
id=virtio-disk0
Update the qemuxml2argvtest with a couple of examples. One for a
simple case and the other a bit more complex where multiple VxHS disks
are added where at least one uses a VxHS that doesn't require TLS
credentials and thus sets the domain disk source attribute "tls = 'no'".
Update the hotplug to be able to handle processing the tlsAlias whether
it's to add the TLS object when hotplugging a disk or to remove the TLS
object when hot unplugging a disk. The hot plug/unplug code is largely
generic, but the addition code does make the VXHS specific checks only
because it needs to grab the correct config directory and generate the
object as the command line would do.
Signed-off-by: Ashish Mittal <Ashish.Mittal@veritas.com>
Signed-off-by: John Ferlan <jferlan@redhat.com>
2017-08-30 15:06:00 +00:00
|
|
|
|
2017-10-18 15:22:23 +00:00
|
|
|
return 0;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-18 08:58:01 +00:00
|
|
|
static int
|
2018-02-09 16:14:41 +00:00
|
|
|
qemuDomainAttachUSBMassStorageDevice(virQEMUDriverPtr driver,
|
2013-07-18 08:58:01 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2015-08-12 14:52:19 +00:00
|
|
|
|
2017-10-20 11:28:21 +00:00
|
|
|
if (virDomainUSBAddressEnsure(priv->usbaddrs, &disk->info) < 0)
|
|
|
|
return -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
if (qemuDomainAttachDiskGeneric(driver, vm, disk) < 0) {
|
2015-08-12 14:52:19 +00:00
|
|
|
virDomainUSBAddressRelease(priv->usbaddrs, &disk->info);
|
2017-10-18 15:27:08 +00:00
|
|
|
return -1;
|
2016-06-29 20:32:58 +00:00
|
|
|
}
|
2016-04-25 16:26:48 +00:00
|
|
|
|
2017-10-18 15:27:08 +00:00
|
|
|
return 0;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-25 12:23:08 +00:00
|
|
|
static int
|
|
|
|
qemuDomainAttachDeviceDiskLiveInternal(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDeviceDefPtr dev)
|
2013-07-18 08:58:01 +00:00
|
|
|
{
|
2016-02-03 09:09:24 +00:00
|
|
|
size_t i;
|
2013-07-18 08:58:01 +00:00
|
|
|
virDomainDiskDefPtr disk = dev->data.disk;
|
|
|
|
int ret = -1;
|
|
|
|
|
2018-09-25 12:47:36 +00:00
|
|
|
if (disk->device == VIR_DOMAIN_DISK_DEVICE_CDROM ||
|
|
|
|
disk->device == VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("cdrom/floppy device hotplug isn't supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-02-09 16:06:43 +00:00
|
|
|
if (virDomainDiskTranslateSourcePool(disk) < 0)
|
2016-02-02 13:59:24 +00:00
|
|
|
goto cleanup;
|
2013-07-18 08:58:01 +00:00
|
|
|
|
|
|
|
if (qemuAddSharedDevice(driver, dev, vm->def->name) < 0)
|
2016-02-02 13:59:24 +00:00
|
|
|
goto cleanup;
|
2013-07-18 08:58:01 +00:00
|
|
|
|
|
|
|
if (qemuSetUnprivSGIO(dev) < 0)
|
2016-02-02 13:59:24 +00:00
|
|
|
goto cleanup;
|
2013-07-18 08:58:01 +00:00
|
|
|
|
2019-01-16 14:33:07 +00:00
|
|
|
if (qemuDomainDetermineDiskChain(driver, vm, disk, NULL, true) < 0)
|
2016-02-02 13:59:24 +00:00
|
|
|
goto cleanup;
|
2013-07-18 08:58:01 +00:00
|
|
|
|
2018-09-25 13:07:01 +00:00
|
|
|
for (i = 0; i < vm->def->ndisks; i++) {
|
|
|
|
if (virDomainDiskDefCheckDuplicateInfo(vm->def->disks[i], disk) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2016-02-02 09:09:33 +00:00
|
|
|
|
2018-09-25 13:07:01 +00:00
|
|
|
switch ((virDomainDiskBus) disk->bus) {
|
|
|
|
case VIR_DOMAIN_DISK_BUS_USB:
|
|
|
|
if (disk->device == VIR_DOMAIN_DISK_DEVICE_LUN) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("disk device='lun' is not supported for usb bus"));
|
2016-02-02 09:09:33 +00:00
|
|
|
break;
|
2013-07-18 08:58:01 +00:00
|
|
|
}
|
2018-09-25 13:07:01 +00:00
|
|
|
ret = qemuDomainAttachUSBMassStorageDevice(driver, vm, disk);
|
2013-07-18 08:58:01 +00:00
|
|
|
break;
|
2016-02-02 08:43:36 +00:00
|
|
|
|
2018-09-25 13:07:01 +00:00
|
|
|
case VIR_DOMAIN_DISK_BUS_VIRTIO:
|
|
|
|
ret = qemuDomainAttachVirtioDiskDevice(driver, vm, disk);
|
2013-07-18 08:58:01 +00:00
|
|
|
break;
|
2018-09-25 13:07:01 +00:00
|
|
|
|
|
|
|
case VIR_DOMAIN_DISK_BUS_SCSI:
|
|
|
|
ret = qemuDomainAttachSCSIDisk(driver, vm, disk);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DISK_BUS_IDE:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_FDC:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_XEN:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_UML:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_SATA:
|
|
|
|
case VIR_DOMAIN_DISK_BUS_SD:
|
|
|
|
/* Note that SD card hotplug support should be added only once
|
|
|
|
* they support '-device' (don't require -drive only).
|
|
|
|
* See also: qemuDiskBusNeedsDriveArg */
|
|
|
|
case VIR_DOMAIN_DISK_BUS_LAST:
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("disk bus '%s' cannot be hotplugged."),
|
|
|
|
virDomainDiskBusTypeToString(disk->bus));
|
2013-07-18 08:58:01 +00:00
|
|
|
}
|
|
|
|
|
2016-02-02 13:59:24 +00:00
|
|
|
cleanup:
|
2013-07-18 08:58:01 +00:00
|
|
|
if (ret != 0)
|
|
|
|
ignore_value(qemuRemoveSharedDevice(driver, dev, vm->def->name));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-25 12:23:08 +00:00
|
|
|
/**
|
|
|
|
* qemuDomainAttachDeviceDiskLive:
|
|
|
|
* @driver: qemu driver struct
|
|
|
|
* @vm: domain object
|
|
|
|
* @dev: device to attach (expected type is DISK)
|
|
|
|
*
|
|
|
|
* Attach a new disk or in case of cdroms/floppies change the media in the drive.
|
|
|
|
* This function handles all the necessary steps to attach a new storage source
|
|
|
|
* to the VM.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
qemuDomainAttachDeviceDiskLive(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDeviceDefPtr dev)
|
|
|
|
{
|
2018-09-25 12:47:36 +00:00
|
|
|
virDomainDiskDefPtr disk = dev->data.disk;
|
|
|
|
virDomainDiskDefPtr orig_disk = NULL;
|
|
|
|
|
|
|
|
/* this API overloads media change semantics on disk hotplug
|
|
|
|
* for devices supporting media changes */
|
|
|
|
if ((disk->device == VIR_DOMAIN_DISK_DEVICE_CDROM ||
|
|
|
|
disk->device == VIR_DOMAIN_DISK_DEVICE_FLOPPY) &&
|
|
|
|
(orig_disk = virDomainDiskFindByBusAndDst(vm->def, disk->bus, disk->dst))) {
|
|
|
|
if (qemuDomainChangeEjectableMedia(driver, vm, orig_disk,
|
|
|
|
disk->src, false) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
disk->src = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-09-25 12:23:08 +00:00
|
|
|
return qemuDomainAttachDeviceDiskLiveInternal(driver, vm, dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-06-28 10:56:47 +00:00
|
|
|
static void
|
|
|
|
qemuDomainNetDeviceVportRemove(virDomainNetDefPtr net)
|
|
|
|
{
|
|
|
|
virNetDevVPortProfilePtr vport = virDomainNetGetActualVirtPortProfile(net);
|
|
|
|
const char *brname;
|
|
|
|
|
|
|
|
if (!vport)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (vport->virtPortType == VIR_NETDEV_VPORT_PROFILE_MIDONET) {
|
|
|
|
ignore_value(virNetDevMidonetUnbindPort(vport));
|
|
|
|
} else if (vport->virtPortType == VIR_NETDEV_VPORT_PROFILE_OPENVSWITCH) {
|
|
|
|
brname = virDomainNetGetActualBridgeName(net);
|
|
|
|
ignore_value(virNetDevOpenvswitchRemovePort(brname, net->ifname));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-06 14:41:33 +00:00
|
|
|
int
|
|
|
|
qemuDomainAttachNetDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainNetDefPtr net)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-09-07 16:29:30 +00:00
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_NET, { .net = net } };
|
2016-10-25 10:16:36 +00:00
|
|
|
virErrorPtr originalError = NULL;
|
2013-05-21 13:50:09 +00:00
|
|
|
char **tapfdName = NULL;
|
|
|
|
int *tapfd = NULL;
|
2014-11-12 15:42:02 +00:00
|
|
|
size_t tapfdSize = 0;
|
2013-05-21 13:50:09 +00:00
|
|
|
char **vhostfdName = NULL;
|
|
|
|
int *vhostfd = NULL;
|
2014-11-12 15:42:02 +00:00
|
|
|
size_t vhostfdSize = 0;
|
2016-12-01 06:01:18 +00:00
|
|
|
size_t queueSize = 0;
|
2010-12-16 16:10:54 +00:00
|
|
|
char *nicstr = NULL;
|
|
|
|
char *netstr = NULL;
|
|
|
|
int ret = -1;
|
2011-04-26 03:40:01 +00:00
|
|
|
bool releaseaddr = false;
|
2011-07-04 06:27:12 +00:00
|
|
|
bool iface_connected = false;
|
2016-09-23 15:04:53 +00:00
|
|
|
virDomainNetType actualType;
|
2015-01-07 14:52:21 +00:00
|
|
|
virNetDevBandwidthPtr actualBandwidth;
|
2013-01-10 21:03:14 +00:00
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2016-07-23 01:47:10 +00:00
|
|
|
virDomainCCWAddressSetPtr ccwaddrs = NULL;
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
size_t i;
|
2016-08-15 16:01:55 +00:00
|
|
|
char *charDevAlias = NULL;
|
|
|
|
bool charDevPlugged = false;
|
|
|
|
bool netdevPlugged = false;
|
2018-03-28 23:25:00 +00:00
|
|
|
char *netdev_name;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
qemu: support type=hostdev network device live hotplug attach/detach
qemuDomainAttachNetDevice
- re-ordered some things at start of function because
networkAllocateActualDevice should always be run and a slot
in def->nets always allocated, but host_net_add isn't needed
if the actual type is hostdev.
- if actual type is hostdev, defer to
qemuDomainAttachHostDevice (which will reach up to the NetDef
for things like MAC address when necessary). After return
from qemuDomainAttachHostDevice, slip directly to cleanup,
since the rest of the function is specific to emulated net
devices.
- put assignment of new NetDef into expanded def->nets down
below cleanup: (but only on success) since it is also needed
for emulated and hostdev net devices.
qemuDomainDetachHostDevice
- after locating the exact device to detach, check if it's a
network device and, if so, use toplevel
qemuDomainDetachNetDevice instead so that the def->nets list
is properly updated, and 'actual device' properly returned to
network pool if appropriate. Otherwise, for normal hostdevs,
call the lower level qemuDomainDetachThisDevice.
qemuDomainDetachNetDevice
- This is where it gets a bit tricky. After locating the device
on the def->nets list, if the network device type == hostdev,
call the *lower level* qemuDomainDetachThisDevice (which will
reach back up to the parent net device for MAC address /
virtualport when appropriate, then clear the device out of
def->hostdevs) before skipping past all the emulated
net-device-specific code to cleanup:, where the network
device is removed from def->nets, and the network device
object is freed.
In short, any time a hostdev-type network device is detached, we must
go through the toplevel virDomaineDetachNetDevice function first and
last, to make sure 1) the def->nnets list is properly managed, and 2)
any device allocated with networkAllocateActualDevice is properly
freed. At the same time, in the middle we need to go through the
lower-level vidDomainDetach*This*HostDevice to be sure that 1) the
def->hostdevs list is properly managed, 2) the PCI device is properly
detached from the guest and reattached to the host (if appropriate),
and 3) any higher level teardown is called at the appropriate time, by
reaching back up to the NetDef config (part (3) will be covered in a
separate patch).
2012-02-27 19:20:17 +00:00
|
|
|
/* preallocate new slot for device */
|
2013-08-28 12:56:21 +00:00
|
|
|
if (VIR_REALLOC_N(vm->def->nets, vm->def->nnets + 1) < 0)
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2011-07-04 06:27:12 +00:00
|
|
|
/* If appropriate, grab a physical device from the configured
|
|
|
|
* network's pool of devices, or resolve bridge device name
|
|
|
|
* to the one defined in the network definition.
|
|
|
|
*/
|
2018-01-25 09:35:47 +00:00
|
|
|
if (virDomainNetAllocateActualDevice(vm->def, net) < 0)
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2011-07-04 06:27:12 +00:00
|
|
|
|
|
|
|
actualType = virDomainNetGetActualType(net);
|
qemu: support type=hostdev network device live hotplug attach/detach
qemuDomainAttachNetDevice
- re-ordered some things at start of function because
networkAllocateActualDevice should always be run and a slot
in def->nets always allocated, but host_net_add isn't needed
if the actual type is hostdev.
- if actual type is hostdev, defer to
qemuDomainAttachHostDevice (which will reach up to the NetDef
for things like MAC address when necessary). After return
from qemuDomainAttachHostDevice, slip directly to cleanup,
since the rest of the function is specific to emulated net
devices.
- put assignment of new NetDef into expanded def->nets down
below cleanup: (but only on success) since it is also needed
for emulated and hostdev net devices.
qemuDomainDetachHostDevice
- after locating the exact device to detach, check if it's a
network device and, if so, use toplevel
qemuDomainDetachNetDevice instead so that the def->nets list
is properly updated, and 'actual device' properly returned to
network pool if appropriate. Otherwise, for normal hostdevs,
call the lower level qemuDomainDetachThisDevice.
qemuDomainDetachNetDevice
- This is where it gets a bit tricky. After locating the device
on the def->nets list, if the network device type == hostdev,
call the *lower level* qemuDomainDetachThisDevice (which will
reach back up to the parent net device for MAC address /
virtualport when appropriate, then clear the device out of
def->hostdevs) before skipping past all the emulated
net-device-specific code to cleanup:, where the network
device is removed from def->nets, and the network device
object is freed.
In short, any time a hostdev-type network device is detached, we must
go through the toplevel virDomaineDetachNetDevice function first and
last, to make sure 1) the def->nnets list is properly managed, and 2)
any device allocated with networkAllocateActualDevice is properly
freed. At the same time, in the middle we need to go through the
lower-level vidDomainDetach*This*HostDevice to be sure that 1) the
def->hostdevs list is properly managed, 2) the PCI device is properly
detached from the guest and reattached to the host (if appropriate),
and 3) any higher level teardown is called at the appropriate time, by
reaching back up to the NetDef config (part (3) will be covered in a
separate patch).
2012-02-27 19:20:17 +00:00
|
|
|
|
2016-02-26 07:02:46 +00:00
|
|
|
/* Currently only TAP/macvtap devices supports multiqueue. */
|
2013-04-18 08:47:01 +00:00
|
|
|
if (net->driver.virtio.queues > 0 &&
|
|
|
|
!(actualType == VIR_DOMAIN_NET_TYPE_NETWORK ||
|
2016-02-26 07:02:46 +00:00
|
|
|
actualType == VIR_DOMAIN_NET_TYPE_BRIDGE ||
|
2016-03-23 11:37:59 +00:00
|
|
|
actualType == VIR_DOMAIN_NET_TYPE_DIRECT ||
|
2016-10-25 10:18:23 +00:00
|
|
|
actualType == VIR_DOMAIN_NET_TYPE_ETHERNET ||
|
|
|
|
actualType == VIR_DOMAIN_NET_TYPE_VHOSTUSER)) {
|
2013-04-18 08:47:01 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("Multiqueue network is not supported for: %s"),
|
|
|
|
virDomainNetTypeToString(actualType));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-08-10 06:05:29 +00:00
|
|
|
/* and only TAP devices support nwfilter rules */
|
|
|
|
if (net->filter &&
|
|
|
|
!(actualType == VIR_DOMAIN_NET_TYPE_NETWORK ||
|
2016-03-23 11:37:59 +00:00
|
|
|
actualType == VIR_DOMAIN_NET_TYPE_BRIDGE ||
|
|
|
|
actualType == VIR_DOMAIN_NET_TYPE_ETHERNET)) {
|
2015-08-10 06:05:29 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("filterref is not supported for "
|
|
|
|
"network interfaces of type %s"),
|
|
|
|
virDomainNetTypeToString(actualType));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-08-15 16:01:55 +00:00
|
|
|
if (qemuAssignDeviceNetAlias(vm->def, net, -1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-12-17 11:30:34 +00:00
|
|
|
if (qemuDomainIsS390CCW(vm->def) &&
|
|
|
|
net->info.type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI &&
|
|
|
|
virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_CCW)) {
|
|
|
|
net->info.type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_CCW;
|
|
|
|
if (!(ccwaddrs = virDomainCCWAddressSetCreateFromDomain(vm->def)))
|
|
|
|
goto cleanup;
|
|
|
|
if (virDomainCCWAddressAssign(&net->info, ccwaddrs,
|
|
|
|
!net->info.addr.ccw.assigned) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
} else if (virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_VIRTIO_S390)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("virtio-s390 net device cannot be hotplugged."));
|
|
|
|
goto cleanup;
|
|
|
|
} else if (qemuDomainEnsurePCIAddress(vm, &dev, driver) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
releaseaddr = true;
|
|
|
|
|
2016-09-26 09:53:58 +00:00
|
|
|
switch (actualType) {
|
|
|
|
case VIR_DOMAIN_NET_TYPE_BRIDGE:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_NETWORK:
|
2013-04-18 08:47:01 +00:00
|
|
|
tapfdSize = vhostfdSize = net->driver.virtio.queues;
|
|
|
|
if (!tapfdSize)
|
|
|
|
tapfdSize = vhostfdSize = 1;
|
2016-12-01 06:01:18 +00:00
|
|
|
queueSize = tapfdSize;
|
2014-03-10 22:47:19 +00:00
|
|
|
if (VIR_ALLOC_N(tapfd, tapfdSize) < 0)
|
2013-05-21 13:50:09 +00:00
|
|
|
goto cleanup;
|
2014-03-10 22:47:19 +00:00
|
|
|
memset(tapfd, -1, sizeof(*tapfd) * tapfdSize);
|
|
|
|
if (VIR_ALLOC_N(vhostfd, vhostfdSize) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
memset(vhostfd, -1, sizeof(*vhostfd) * vhostfdSize);
|
2016-02-15 15:52:50 +00:00
|
|
|
if (qemuInterfaceBridgeConnect(vm->def, driver, net,
|
2017-05-18 18:16:27 +00:00
|
|
|
tapfd, &tapfdSize) < 0)
|
2013-04-20 09:11:25 +00:00
|
|
|
goto cleanup;
|
|
|
|
iface_connected = true;
|
2018-03-29 10:51:55 +00:00
|
|
|
if (qemuInterfaceOpenVhostNet(vm->def, net, vhostfd, &vhostfdSize) < 0)
|
2013-04-20 09:11:25 +00:00
|
|
|
goto cleanup;
|
2016-09-26 09:53:58 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_DIRECT:
|
2015-12-04 10:47:22 +00:00
|
|
|
tapfdSize = vhostfdSize = net->driver.virtio.queues;
|
|
|
|
if (!tapfdSize)
|
|
|
|
tapfdSize = vhostfdSize = 1;
|
2016-12-01 06:01:18 +00:00
|
|
|
queueSize = tapfdSize;
|
2015-12-04 10:47:22 +00:00
|
|
|
if (VIR_ALLOC_N(tapfd, tapfdSize) < 0)
|
2014-03-10 22:47:19 +00:00
|
|
|
goto cleanup;
|
2015-12-04 10:47:22 +00:00
|
|
|
memset(tapfd, -1, sizeof(*tapfd) * tapfdSize);
|
|
|
|
if (VIR_ALLOC_N(vhostfd, vhostfdSize) < 0)
|
2013-05-21 13:50:09 +00:00
|
|
|
goto cleanup;
|
2015-12-04 10:47:22 +00:00
|
|
|
memset(vhostfd, -1, sizeof(*vhostfd) * vhostfdSize);
|
2016-02-15 15:26:40 +00:00
|
|
|
if (qemuInterfaceDirectConnect(vm->def, driver, net,
|
|
|
|
tapfd, tapfdSize,
|
|
|
|
VIR_NETDEV_VPORT_PROFILE_OP_CREATE) < 0)
|
2011-07-04 06:27:12 +00:00
|
|
|
goto cleanup;
|
|
|
|
iface_connected = true;
|
2018-03-29 10:51:55 +00:00
|
|
|
if (qemuInterfaceOpenVhostNet(vm->def, net, vhostfd, &vhostfdSize) < 0)
|
2011-03-09 04:43:33 +00:00
|
|
|
goto cleanup;
|
2016-09-26 09:53:58 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_ETHERNET:
|
2016-03-23 11:37:59 +00:00
|
|
|
tapfdSize = vhostfdSize = net->driver.virtio.queues;
|
|
|
|
if (!tapfdSize)
|
|
|
|
tapfdSize = vhostfdSize = 1;
|
2016-12-01 06:01:18 +00:00
|
|
|
queueSize = tapfdSize;
|
2016-03-23 11:37:59 +00:00
|
|
|
if (VIR_ALLOC_N(tapfd, tapfdSize) < 0)
|
2013-05-21 13:50:09 +00:00
|
|
|
goto cleanup;
|
2016-03-23 11:37:59 +00:00
|
|
|
memset(tapfd, -1, sizeof(*tapfd) * tapfdSize);
|
|
|
|
if (VIR_ALLOC_N(vhostfd, vhostfdSize) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
memset(vhostfd, -1, sizeof(*vhostfd) * vhostfdSize);
|
|
|
|
if (qemuInterfaceEthernetConnect(vm->def, driver, net,
|
2016-09-26 09:53:58 +00:00
|
|
|
tapfd, tapfdSize) < 0)
|
2016-03-23 11:37:59 +00:00
|
|
|
goto cleanup;
|
|
|
|
iface_connected = true;
|
2018-03-29 10:51:55 +00:00
|
|
|
if (qemuInterfaceOpenVhostNet(vm->def, net, vhostfd, &vhostfdSize) < 0)
|
2013-02-08 17:19:09 +00:00
|
|
|
goto cleanup;
|
2016-09-26 09:53:58 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_HOSTDEV:
|
2016-09-26 09:41:36 +00:00
|
|
|
/* This is really a "smart hostdev", so it should be attached
|
|
|
|
* as a hostdev (the hostdev code will reach over into the
|
|
|
|
* netdev-specific code as appropriate), then also added to
|
|
|
|
* the nets list (see cleanup:) if successful.
|
|
|
|
*/
|
2018-02-09 16:14:41 +00:00
|
|
|
ret = qemuDomainAttachHostDevice(driver, vm,
|
2016-09-26 09:41:36 +00:00
|
|
|
virDomainNetGetActualHostdev(net));
|
|
|
|
goto cleanup;
|
2016-09-26 09:53:58 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_VHOSTUSER:
|
2016-12-01 06:01:18 +00:00
|
|
|
queueSize = net->driver.virtio.queues;
|
|
|
|
if (!queueSize)
|
|
|
|
queueSize = 1;
|
2018-03-28 23:36:20 +00:00
|
|
|
if (!qemuDomainSupportsNicdev(vm->def, net)) {
|
2016-08-15 16:01:55 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2018-03-28 23:36:20 +00:00
|
|
|
"%s", _("Nicdev support unavailable"));
|
2016-08-15 16:01:55 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2016-10-18 14:37:23 +00:00
|
|
|
if (!(charDevAlias = qemuAliasChardevFromDevAlias(net->info.alias)))
|
2016-08-15 16:01:55 +00:00
|
|
|
goto cleanup;
|
2018-09-18 08:59:05 +00:00
|
|
|
|
|
|
|
if (virNetDevOpenvswitchGetVhostuserIfname(net->data.vhostuser->data.nix.path,
|
|
|
|
&net->ifname) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-08-15 16:01:55 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_USER:
|
2017-02-28 16:49:49 +00:00
|
|
|
/* No preparation needed. */
|
|
|
|
break;
|
|
|
|
|
2016-09-26 09:53:58 +00:00
|
|
|
case VIR_DOMAIN_NET_TYPE_SERVER:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_CLIENT:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_MCAST:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_INTERNAL:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_UDP:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_LAST:
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("hotplug of interface type of %s is not implemented yet"),
|
|
|
|
virDomainNetTypeToString(actualType));
|
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2014-09-16 20:50:53 +00:00
|
|
|
/* Set device online immediately */
|
|
|
|
if (qemuInterfaceStartDevice(net) < 0)
|
2015-01-06 09:20:30 +00:00
|
|
|
goto cleanup;
|
2014-09-16 20:50:53 +00:00
|
|
|
|
2015-01-07 14:52:21 +00:00
|
|
|
/* Set bandwidth or warn if requested and not supported. */
|
|
|
|
actualBandwidth = virDomainNetGetActualBandwidth(net);
|
|
|
|
if (actualBandwidth) {
|
|
|
|
if (virNetDevSupportBandwidth(actualType)) {
|
2017-10-02 12:12:44 +00:00
|
|
|
if (virNetDevBandwidthSet(net->ifname, actualBandwidth, false,
|
|
|
|
!virDomainNetTypeSharesHostView(net)) < 0)
|
2015-01-07 14:52:21 +00:00
|
|
|
goto cleanup;
|
|
|
|
} else {
|
|
|
|
VIR_WARN("setting bandwidth on interfaces of "
|
|
|
|
"type '%s' is not implemented yet",
|
|
|
|
virDomainNetTypeToString(actualType));
|
|
|
|
}
|
|
|
|
}
|
2014-11-18 23:55:48 +00:00
|
|
|
|
2017-06-08 08:14:36 +00:00
|
|
|
if (net->mtu &&
|
|
|
|
virNetDevSetMTU(net->ifname, net->mtu) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2014-08-13 14:08:03 +00:00
|
|
|
for (i = 0; i < tapfdSize; i++) {
|
2017-02-13 13:36:53 +00:00
|
|
|
if (qemuSecuritySetTapFDLabel(driver->securityManager,
|
|
|
|
vm->def, tapfd[i]) < 0)
|
2014-08-13 14:08:03 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2013-05-21 13:50:09 +00:00
|
|
|
if (VIR_ALLOC_N(tapfdName, tapfdSize) < 0 ||
|
2013-07-04 10:14:12 +00:00
|
|
|
VIR_ALLOC_N(vhostfdName, vhostfdSize) < 0)
|
2013-05-21 13:50:09 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
for (i = 0; i < tapfdSize; i++) {
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
if (virAsprintf(&tapfdName[i], "fd-%s%zu", net->info.alias, i) < 0)
|
2013-07-04 10:14:12 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2013-05-21 13:50:09 +00:00
|
|
|
for (i = 0; i < vhostfdSize; i++) {
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
if (virAsprintf(&vhostfdName[i], "vhostfd-%s%zu", net->info.alias, i) < 0)
|
2013-07-04 10:14:12 +00:00
|
|
|
goto cleanup;
|
2011-03-09 04:43:33 +00:00
|
|
|
}
|
|
|
|
|
2018-03-28 21:36:13 +00:00
|
|
|
if (!(netstr = qemuBuildHostNetStr(net, driver,
|
|
|
|
tapfdName, tapfdSize,
|
|
|
|
vhostfdName, vhostfdSize)))
|
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-02-06 18:17:20 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2016-08-15 16:01:55 +00:00
|
|
|
|
|
|
|
if (actualType == VIR_DOMAIN_NET_TYPE_VHOSTUSER) {
|
|
|
|
if (qemuMonitorAttachCharDev(priv->mon, charDevAlias, net->data.vhostuser) < 0) {
|
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
|
|
|
virDomainAuditNet(vm, NULL, net, "attach", false);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
charDevPlugged = true;
|
|
|
|
}
|
|
|
|
|
2018-03-28 21:36:13 +00:00
|
|
|
if (qemuMonitorAddNetdev(priv->mon, netstr,
|
|
|
|
tapfd, tapfdName, tapfdSize,
|
|
|
|
vhostfd, vhostfdName, vhostfdSize) < 0) {
|
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
|
|
|
virDomainAuditNet(vm, NULL, net, "attach", false);
|
|
|
|
goto try_remove;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
2018-03-28 21:36:13 +00:00
|
|
|
netdevPlugged = true;
|
2016-08-15 16:01:55 +00:00
|
|
|
|
2014-12-16 09:40:58 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-05-21 13:50:09 +00:00
|
|
|
for (i = 0; i < tapfdSize; i++)
|
|
|
|
VIR_FORCE_CLOSE(tapfd[i]);
|
|
|
|
for (i = 0; i < vhostfdSize; i++)
|
|
|
|
VIR_FORCE_CLOSE(vhostfd[i]);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2018-06-19 15:42:37 +00:00
|
|
|
if (!(nicstr = qemuBuildNicDevStr(vm->def, net, 0,
|
2017-05-18 18:16:27 +00:00
|
|
|
queueSize, priv->qemuCaps)))
|
2016-04-26 12:04:49 +00:00
|
|
|
goto try_remove;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-02-06 18:17:20 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2018-11-08 11:00:30 +00:00
|
|
|
|
|
|
|
if (qemuDomainAttachExtensionDevice(priv->mon, &net->info) < 0) {
|
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
|
|
|
virDomainAuditNet(vm, NULL, net, "attach", false);
|
|
|
|
goto try_remove;
|
|
|
|
}
|
|
|
|
|
2016-04-26 12:04:49 +00:00
|
|
|
if (qemuMonitorAddDevice(priv->mon, nicstr) < 0) {
|
2018-11-08 11:00:30 +00:00
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, &net->info));
|
2016-04-26 12:04:49 +00:00
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
|
|
|
virDomainAuditNet(vm, NULL, net, "attach", false);
|
|
|
|
goto try_remove;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
2014-12-16 09:40:58 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2011-09-06 08:23:47 +00:00
|
|
|
/* set link state */
|
|
|
|
if (net->linkstate == VIR_DOMAIN_NET_INTERFACE_LINK_STATE_DOWN) {
|
|
|
|
if (!net->info.alias) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("device alias not found: cannot set link state to down"));
|
2011-09-06 08:23:47 +00:00
|
|
|
} else {
|
2013-02-06 18:17:20 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2011-09-06 08:23:47 +00:00
|
|
|
|
2018-03-28 21:36:13 +00:00
|
|
|
if (qemuMonitorSetLink(priv->mon, net->info.alias, VIR_DOMAIN_NET_INTERFACE_LINK_STATE_DOWN) < 0) {
|
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
|
|
|
virDomainAuditNet(vm, NULL, net, "attach", false);
|
|
|
|
goto try_remove;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
|
|
|
|
2014-12-16 09:40:58 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
|
|
|
/* link set to down */
|
|
|
|
}
|
|
|
|
|
Move qemu_audit.h helpers into shared code
The LXC and UML drivers can both make use of auditing. Move
the qemu_audit.{c,h} files to src/conf/domain_audit.{c,h}
* src/conf/domain_audit.c: Rename from src/qemu/qemu_audit.c
* src/conf/domain_audit.h: Rename from src/qemu/qemu_audit.h
* src/Makefile.am: Remove qemu_audit.{c,h}, add domain_audit.{c,h}
* src/qemu/qemu_audit.h, src/qemu/qemu_cgroup.c,
src/qemu/qemu_command.c, src/qemu/qemu_driver.c,
src/qemu/qemu_hotplug.c, src/qemu/qemu_migration.c,
src/qemu/qemu_process.c: Update for changed audit API names
2011-07-04 10:56:13 +00:00
|
|
|
virDomainAuditNet(vm, NULL, net, "attach", true);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
qemu: support type=hostdev network device live hotplug attach/detach
qemuDomainAttachNetDevice
- re-ordered some things at start of function because
networkAllocateActualDevice should always be run and a slot
in def->nets always allocated, but host_net_add isn't needed
if the actual type is hostdev.
- if actual type is hostdev, defer to
qemuDomainAttachHostDevice (which will reach up to the NetDef
for things like MAC address when necessary). After return
from qemuDomainAttachHostDevice, slip directly to cleanup,
since the rest of the function is specific to emulated net
devices.
- put assignment of new NetDef into expanded def->nets down
below cleanup: (but only on success) since it is also needed
for emulated and hostdev net devices.
qemuDomainDetachHostDevice
- after locating the exact device to detach, check if it's a
network device and, if so, use toplevel
qemuDomainDetachNetDevice instead so that the def->nets list
is properly updated, and 'actual device' properly returned to
network pool if appropriate. Otherwise, for normal hostdevs,
call the lower level qemuDomainDetachThisDevice.
qemuDomainDetachNetDevice
- This is where it gets a bit tricky. After locating the device
on the def->nets list, if the network device type == hostdev,
call the *lower level* qemuDomainDetachThisDevice (which will
reach back up to the parent net device for MAC address /
virtualport when appropriate, then clear the device out of
def->hostdevs) before skipping past all the emulated
net-device-specific code to cleanup:, where the network
device is removed from def->nets, and the network device
object is freed.
In short, any time a hostdev-type network device is detached, we must
go through the toplevel virDomaineDetachNetDevice function first and
last, to make sure 1) the def->nnets list is properly managed, and 2)
any device allocated with networkAllocateActualDevice is properly
freed. At the same time, in the middle we need to go through the
lower-level vidDomainDetach*This*HostDevice to be sure that 1) the
def->hostdevs list is properly managed, 2) the PCI device is properly
detached from the guest and reattached to the host (if appropriate),
and 3) any higher level teardown is called at the appropriate time, by
reaching back up to the NetDef config (part (3) will be covered in a
separate patch).
2012-02-27 19:20:17 +00:00
|
|
|
if (!ret) {
|
|
|
|
vm->def->nets[vm->def->nnets++] = net;
|
|
|
|
} else {
|
2013-07-09 20:30:57 +00:00
|
|
|
if (releaseaddr)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &net->info, NULL);
|
2011-07-04 06:27:12 +00:00
|
|
|
|
2012-02-28 03:43:23 +00:00
|
|
|
if (iface_connected) {
|
2018-08-21 13:28:18 +00:00
|
|
|
virErrorPreserveLast(&originalError);
|
2011-07-04 06:27:12 +00:00
|
|
|
virDomainConfNWFilterTeardown(net);
|
2018-08-21 13:28:18 +00:00
|
|
|
virErrorRestore(&originalError);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-07-01 15:04:57 +00:00
|
|
|
if (virDomainNetGetActualType(net) == VIR_DOMAIN_NET_TYPE_DIRECT) {
|
|
|
|
ignore_value(virNetDevMacVLanDeleteWithVPortProfile(
|
|
|
|
net->ifname, &net->mac,
|
|
|
|
virDomainNetGetActualDirectDev(net),
|
|
|
|
virDomainNetGetActualDirectMode(net),
|
|
|
|
virDomainNetGetActualVirtPortProfile(net),
|
|
|
|
cfg->stateDir));
|
|
|
|
}
|
|
|
|
|
2017-06-28 10:56:47 +00:00
|
|
|
qemuDomainNetDeviceVportRemove(net);
|
2012-02-28 03:43:23 +00:00
|
|
|
}
|
2012-02-10 21:09:00 +00:00
|
|
|
|
2013-08-27 17:06:18 +00:00
|
|
|
virDomainNetRemoveHostdev(vm->def, net);
|
|
|
|
|
2018-01-25 09:35:47 +00:00
|
|
|
virDomainNetReleaseActualDevice(vm->def, net);
|
2011-07-04 06:27:12 +00:00
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
VIR_FREE(nicstr);
|
|
|
|
VIR_FREE(netstr);
|
2013-04-18 08:47:01 +00:00
|
|
|
for (i = 0; tapfd && i < tapfdSize; i++) {
|
2013-05-21 13:50:09 +00:00
|
|
|
VIR_FORCE_CLOSE(tapfd[i]);
|
2013-04-18 08:47:01 +00:00
|
|
|
if (tapfdName)
|
|
|
|
VIR_FREE(tapfdName[i]);
|
2013-05-21 13:50:09 +00:00
|
|
|
}
|
|
|
|
VIR_FREE(tapfd);
|
|
|
|
VIR_FREE(tapfdName);
|
2013-04-18 08:47:01 +00:00
|
|
|
for (i = 0; vhostfd && i < vhostfdSize; i++) {
|
2013-05-21 13:50:09 +00:00
|
|
|
VIR_FORCE_CLOSE(vhostfd[i]);
|
2013-04-18 08:47:01 +00:00
|
|
|
if (vhostfdName)
|
|
|
|
VIR_FREE(vhostfdName[i]);
|
2013-05-21 13:50:09 +00:00
|
|
|
}
|
|
|
|
VIR_FREE(vhostfd);
|
|
|
|
VIR_FREE(vhostfdName);
|
2016-08-15 16:01:55 +00:00
|
|
|
VIR_FREE(charDevAlias);
|
2013-01-10 21:03:14 +00:00
|
|
|
virObjectUnref(cfg);
|
2016-07-23 01:47:10 +00:00
|
|
|
virDomainCCWAddressSetFree(ccwaddrs);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
try_remove:
|
2010-12-16 16:10:54 +00:00
|
|
|
if (!virDomainObjIsActive(vm))
|
|
|
|
goto cleanup;
|
|
|
|
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&originalError);
|
2018-03-28 23:25:00 +00:00
|
|
|
if (virAsprintf(&netdev_name, "host%s", net->info.alias) >= 0) {
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
if (charDevPlugged &&
|
|
|
|
qemuMonitorDetachCharDev(priv->mon, charDevAlias) < 0)
|
|
|
|
VIR_WARN("Failed to remove associated chardev %s", charDevAlias);
|
|
|
|
if (netdevPlugged &&
|
|
|
|
qemuMonitorRemoveNetdev(priv->mon, netdev_name) < 0)
|
|
|
|
VIR_WARN("Failed to remove network backend for netdev %s",
|
|
|
|
netdev_name);
|
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
|
|
|
VIR_FREE(netdev_name);
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&originalError);
|
2010-12-16 16:10:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-05 18:09:04 +00:00
|
|
|
static int
|
2014-03-06 07:35:03 +00:00
|
|
|
qemuDomainAttachHostPCIDevice(virQEMUDriverPtr driver,
|
2013-12-05 18:09:04 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-09-07 16:29:30 +00:00
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_HOSTDEV,
|
|
|
|
{ .hostdev = hostdev } };
|
2017-06-05 13:29:40 +00:00
|
|
|
virDomainDeviceInfoPtr info = hostdev->info;
|
2010-12-16 16:10:54 +00:00
|
|
|
int ret;
|
|
|
|
char *devstr = NULL;
|
|
|
|
int configfd = -1;
|
|
|
|
char *configfd_name = NULL;
|
2011-04-26 03:40:01 +00:00
|
|
|
bool releaseaddr = false;
|
2013-11-14 11:02:40 +00:00
|
|
|
bool teardowncgroup = false;
|
2013-12-05 19:54:41 +00:00
|
|
|
bool teardownlabel = false;
|
2016-11-16 14:27:47 +00:00
|
|
|
bool teardowndevice = false;
|
2014-02-04 11:12:47 +00:00
|
|
|
int backend;
|
2014-03-05 10:56:26 +00:00
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
|
|
|
unsigned int flags = 0;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-08-28 12:56:21 +00:00
|
|
|
if (VIR_REALLOC_N(vm->def->hostdevs, vm->def->nhostdevs + 1) < 0)
|
2016-05-25 08:47:12 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2014-03-05 10:56:26 +00:00
|
|
|
if (!cfg->relaxedACS)
|
|
|
|
flags |= VIR_HOSTDEV_STRICT_ACS_CHECK;
|
2015-10-20 12:10:16 +00:00
|
|
|
if (qemuHostdevPreparePCIDevices(driver, vm->def->name, vm->def->uuid,
|
2014-03-05 10:56:26 +00:00
|
|
|
&hostdev, 1, priv->qemuCaps, flags) < 0)
|
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2015-10-20 12:10:16 +00:00
|
|
|
/* this could have been changed by qemuHostdevPreparePCIDevices */
|
2014-02-04 11:12:47 +00:00
|
|
|
backend = hostdev->source.subsys.u.pci.backend;
|
|
|
|
|
2018-04-25 12:42:34 +00:00
|
|
|
switch ((virDomainHostdevSubsysPCIBackendType)backend) {
|
2013-09-19 14:48:23 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_PCI_BACKEND_VFIO:
|
2013-04-25 16:45:55 +00:00
|
|
|
if (!virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_DEVICE_VFIO_PCI)) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("VFIO PCI device assignment is not "
|
|
|
|
"supported by this version of qemu"));
|
|
|
|
goto error;
|
|
|
|
}
|
2013-09-19 14:48:23 +00:00
|
|
|
break;
|
|
|
|
|
2016-04-26 12:18:04 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_PCI_BACKEND_DEFAULT:
|
|
|
|
case VIR_DOMAIN_HOSTDEV_PCI_BACKEND_KVM:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_HOSTDEV_PCI_BACKEND_XEN:
|
|
|
|
case VIR_DOMAIN_HOSTDEV_PCI_BACKEND_TYPE_LAST:
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("QEMU does not support device assignment mode '%s'"),
|
|
|
|
virDomainHostdevSubsysPCIBackendTypeToString(backend));
|
|
|
|
goto error;
|
2013-09-19 14:48:23 +00:00
|
|
|
break;
|
2013-04-25 11:58:37 +00:00
|
|
|
}
|
|
|
|
|
2015-11-18 11:10:33 +00:00
|
|
|
/* Temporarily add the hostdev to the domain definition. This is needed
|
2015-11-23 16:57:40 +00:00
|
|
|
* because qemuDomainAdjustMaxMemLock() requires the hostdev to be already
|
|
|
|
* part of the domain definition, but other functions like
|
|
|
|
* qemuAssignDeviceHostdevAlias() used below expect it *not* to be there.
|
|
|
|
* A better way to handle this would be nice */
|
2015-11-18 11:10:33 +00:00
|
|
|
vm->def->hostdevs[vm->def->nhostdevs++] = hostdev;
|
2015-11-23 16:57:40 +00:00
|
|
|
if (qemuDomainAdjustMaxMemLock(vm) < 0) {
|
|
|
|
vm->def->hostdevs[--(vm->def->nhostdevs)] = NULL;
|
|
|
|
goto error;
|
2015-11-18 11:10:33 +00:00
|
|
|
}
|
|
|
|
vm->def->hostdevs[--(vm->def->nhostdevs)] = NULL;
|
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceSetupHostdev(vm, hostdev) < 0)
|
2016-11-16 14:27:47 +00:00
|
|
|
goto error;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
2015-11-19 13:35:46 +00:00
|
|
|
if (qemuSetupHostdevCgroup(vm, hostdev) < 0)
|
2013-11-14 11:02:40 +00:00
|
|
|
goto error;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
2016-11-16 14:27:47 +00:00
|
|
|
if (qemuSecuritySetHostdevLabel(driver, vm, hostdev) < 0)
|
2013-12-05 19:54:41 +00:00
|
|
|
goto error;
|
2016-02-05 19:07:50 +00:00
|
|
|
if (backend != VIR_DOMAIN_HOSTDEV_PCI_BACKEND_VFIO)
|
|
|
|
teardownlabel = true;
|
2013-12-05 19:54:41 +00:00
|
|
|
|
2017-06-05 13:29:40 +00:00
|
|
|
if (qemuAssignDeviceHostdevAlias(vm->def, &info->alias, -1) < 0)
|
2016-04-26 12:23:12 +00:00
|
|
|
goto error;
|
2017-06-02 13:19:04 +00:00
|
|
|
|
|
|
|
if (qemuDomainIsPSeries(vm->def)) {
|
|
|
|
/* Isolation groups are only relevant for pSeries guests */
|
|
|
|
if (qemuDomainFillDeviceIsolationGroup(vm->def, &dev) < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2016-11-03 20:33:32 +00:00
|
|
|
if (qemuDomainEnsurePCIAddress(vm, &dev, driver) < 0)
|
2016-04-26 12:23:12 +00:00
|
|
|
goto error;
|
|
|
|
releaseaddr = true;
|
2018-09-07 11:31:23 +00:00
|
|
|
if (backend != VIR_DOMAIN_HOSTDEV_PCI_BACKEND_VFIO) {
|
2016-04-26 12:23:12 +00:00
|
|
|
configfd = qemuOpenPCIConfig(hostdev);
|
|
|
|
if (configfd >= 0) {
|
2017-06-05 13:29:40 +00:00
|
|
|
if (virAsprintf(&configfd_name, "fd-%s", info->alias) < 0)
|
2016-04-26 12:23:12 +00:00
|
|
|
goto error;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
2016-04-26 12:23:12 +00:00
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2016-04-26 12:23:12 +00:00
|
|
|
if (!virDomainObjIsActive(vm)) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("guest unexpectedly quit during hotplug"));
|
|
|
|
goto error;
|
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2016-04-26 12:23:12 +00:00
|
|
|
if (!(devstr = qemuBuildPCIHostdevDevStr(vm->def, hostdev, 0,
|
|
|
|
configfd_name, priv->qemuCaps)))
|
|
|
|
goto error;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2016-04-26 12:23:12 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2018-11-08 11:00:30 +00:00
|
|
|
|
|
|
|
if ((ret = qemuDomainAttachExtensionDevice(priv->mon, hostdev->info)) < 0)
|
|
|
|
goto exit_monitor;
|
|
|
|
|
|
|
|
if ((ret = qemuMonitorAddDeviceWithFd(priv->mon, devstr,
|
|
|
|
configfd, configfd_name)) < 0) {
|
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, hostdev->info));
|
|
|
|
}
|
|
|
|
|
|
|
|
exit_monitor:
|
2016-04-26 12:23:12 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto error;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
Move qemu_audit.h helpers into shared code
The LXC and UML drivers can both make use of auditing. Move
the qemu_audit.{c,h} files to src/conf/domain_audit.{c,h}
* src/conf/domain_audit.c: Rename from src/qemu/qemu_audit.c
* src/conf/domain_audit.h: Rename from src/qemu/qemu_audit.h
* src/Makefile.am: Remove qemu_audit.{c,h}, add domain_audit.{c,h}
* src/qemu/qemu_audit.h, src/qemu/qemu_cgroup.c,
src/qemu/qemu_command.c, src/qemu/qemu_driver.c,
src/qemu/qemu_hotplug.c, src/qemu/qemu_migration.c,
src/qemu/qemu_process.c: Update for changed audit API names
2011-07-04 10:56:13 +00:00
|
|
|
virDomainAuditHostdev(vm, hostdev, "attach", ret == 0);
|
2010-12-16 16:10:54 +00:00
|
|
|
if (ret < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
vm->def->hostdevs[vm->def->nhostdevs++] = hostdev;
|
|
|
|
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
VIR_FREE(configfd_name);
|
|
|
|
VIR_FORCE_CLOSE(configfd);
|
2014-03-05 10:56:26 +00:00
|
|
|
virObjectUnref(cfg);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
error:
|
2013-11-14 11:02:40 +00:00
|
|
|
if (teardowncgroup && qemuTeardownHostdevCgroup(vm, hostdev) < 0)
|
|
|
|
VIR_WARN("Unable to remove host device cgroup ACL on hotplug fail");
|
2013-12-05 19:54:41 +00:00
|
|
|
if (teardownlabel &&
|
2016-11-16 14:27:47 +00:00
|
|
|
qemuSecurityRestoreHostdevLabel(driver, vm, hostdev) < 0)
|
2013-12-05 19:54:41 +00:00
|
|
|
VIR_WARN("Unable to restore host device labelling on hotplug fail");
|
2016-11-16 14:27:47 +00:00
|
|
|
if (teardowndevice &&
|
2017-11-24 16:52:15 +00:00
|
|
|
qemuDomainNamespaceTeardownHostdev(vm, hostdev) < 0)
|
2016-11-16 14:27:47 +00:00
|
|
|
VIR_WARN("Unable to remove host device from /dev");
|
2013-11-14 11:02:40 +00:00
|
|
|
|
2013-07-09 20:30:57 +00:00
|
|
|
if (releaseaddr)
|
2017-06-05 13:29:40 +00:00
|
|
|
qemuDomainReleaseDeviceAddress(vm, info, NULL);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2015-10-20 12:12:48 +00:00
|
|
|
qemuHostdevReAttachPCIDevices(driver, vm->def->name, &hostdev, 1);
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
VIR_FREE(configfd_name);
|
|
|
|
VIR_FORCE_CLOSE(configfd);
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2014-03-05 10:56:26 +00:00
|
|
|
virObjectUnref(cfg);
|
2010-12-16 16:10:54 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-16 15:33:35 +00:00
|
|
|
void
|
|
|
|
qemuDomainDelTLSObjects(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2017-03-09 14:20:27 +00:00
|
|
|
qemuDomainAsyncJob asyncJob,
|
2017-02-16 15:33:35 +00:00
|
|
|
const char *secAlias,
|
|
|
|
const char *tlsAlias)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virErrorPtr orig_err;
|
|
|
|
|
|
|
|
if (!tlsAlias && !secAlias)
|
|
|
|
return;
|
|
|
|
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2017-02-16 15:33:35 +00:00
|
|
|
|
2017-03-09 14:20:27 +00:00
|
|
|
if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0)
|
|
|
|
goto cleanup;
|
2017-02-16 15:33:35 +00:00
|
|
|
|
|
|
|
if (tlsAlias)
|
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
|
|
|
|
|
|
|
if (secAlias)
|
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, secAlias));
|
|
|
|
|
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
|
|
|
|
2017-03-09 14:20:27 +00:00
|
|
|
cleanup:
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2017-02-16 15:33:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainAddTLSObjects(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2017-03-09 14:20:27 +00:00
|
|
|
qemuDomainAsyncJob asyncJob,
|
2017-02-16 15:33:35 +00:00
|
|
|
virJSONValuePtr *secProps,
|
|
|
|
virJSONValuePtr *tlsProps)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virErrorPtr orig_err;
|
2018-05-22 05:38:22 +00:00
|
|
|
char *secAlias = NULL;
|
2017-02-16 15:33:35 +00:00
|
|
|
|
2018-05-22 05:38:22 +00:00
|
|
|
if (!tlsProps && !secProps)
|
2017-02-16 15:33:35 +00:00
|
|
|
return 0;
|
|
|
|
|
2017-03-09 14:20:27 +00:00
|
|
|
if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0)
|
|
|
|
return -1;
|
2017-02-16 15:33:35 +00:00
|
|
|
|
2018-07-04 14:28:58 +00:00
|
|
|
if (secProps && *secProps &&
|
2018-05-22 05:38:22 +00:00
|
|
|
qemuMonitorAddObject(priv->mon, secProps, &secAlias) < 0)
|
|
|
|
goto error;
|
2017-02-16 15:33:35 +00:00
|
|
|
|
2018-05-22 05:38:22 +00:00
|
|
|
if (tlsProps &&
|
|
|
|
qemuMonitorAddObject(priv->mon, tlsProps, NULL) < 0)
|
|
|
|
goto error;
|
2017-02-16 15:33:35 +00:00
|
|
|
|
2018-05-22 05:38:22 +00:00
|
|
|
VIR_FREE(secAlias);
|
|
|
|
|
2017-02-16 15:33:35 +00:00
|
|
|
return qemuDomainObjExitMonitor(driver, vm);
|
|
|
|
|
|
|
|
error:
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2017-02-16 15:33:35 +00:00
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2018-05-22 05:38:22 +00:00
|
|
|
qemuDomainDelTLSObjects(driver, vm, asyncJob, secAlias, NULL);
|
2018-05-22 05:38:22 +00:00
|
|
|
VIR_FREE(secAlias);
|
2017-02-16 15:33:35 +00:00
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-17 15:06:14 +00:00
|
|
|
int
|
|
|
|
qemuDomainGetTLSObjects(virQEMUCapsPtr qemuCaps,
|
|
|
|
qemuDomainSecretInfoPtr secinfo,
|
|
|
|
const char *tlsCertdir,
|
|
|
|
bool tlsListen,
|
|
|
|
bool tlsVerify,
|
2018-05-22 05:38:22 +00:00
|
|
|
const char *alias,
|
2017-02-17 15:06:14 +00:00
|
|
|
virJSONValuePtr *tlsProps,
|
2018-05-29 18:17:04 +00:00
|
|
|
virJSONValuePtr *secProps)
|
2016-10-21 13:38:18 +00:00
|
|
|
{
|
2018-05-29 18:17:04 +00:00
|
|
|
const char *secAlias = NULL;
|
|
|
|
|
2017-02-17 15:06:14 +00:00
|
|
|
if (secinfo) {
|
|
|
|
if (qemuBuildSecretInfoProps(secinfo, secProps) < 0)
|
2016-06-17 13:44:30 +00:00
|
|
|
return -1;
|
|
|
|
|
2018-05-29 18:17:04 +00:00
|
|
|
secAlias = secinfo->s.aes.alias;
|
2016-06-17 13:44:30 +00:00
|
|
|
}
|
|
|
|
|
2017-02-17 15:06:14 +00:00
|
|
|
if (qemuBuildTLSx509BackendProps(tlsCertdir, tlsListen, tlsVerify,
|
2018-05-22 05:38:22 +00:00
|
|
|
alias, secAlias, qemuCaps, tlsProps) < 0)
|
2016-10-21 13:38:18 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-22 17:54:10 +00:00
|
|
|
static int
|
2018-02-09 16:14:41 +00:00
|
|
|
qemuDomainAddChardevTLSObjects(virQEMUDriverPtr driver,
|
2017-02-22 17:54:10 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainChrSourceDefPtr dev,
|
2017-02-17 14:37:34 +00:00
|
|
|
char *devAlias,
|
2017-02-22 17:54:10 +00:00
|
|
|
char *charAlias,
|
|
|
|
char **tlsAlias,
|
2018-05-29 18:03:07 +00:00
|
|
|
const char **secAlias)
|
2017-02-22 17:54:10 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2017-02-17 14:46:16 +00:00
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2017-02-22 17:54:10 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2017-02-17 15:06:14 +00:00
|
|
|
qemuDomainChrSourcePrivatePtr chrSourcePriv;
|
|
|
|
qemuDomainSecretInfoPtr secinfo = NULL;
|
2017-02-22 17:54:10 +00:00
|
|
|
virJSONValuePtr tlsProps = NULL;
|
|
|
|
virJSONValuePtr secProps = NULL;
|
|
|
|
|
2017-02-17 14:46:16 +00:00
|
|
|
/* NB: This may alter haveTLS based on cfg */
|
|
|
|
qemuDomainPrepareChardevSourceTLS(dev, cfg);
|
|
|
|
|
2017-02-22 17:54:10 +00:00
|
|
|
if (dev->type != VIR_DOMAIN_CHR_TYPE_TCP ||
|
2017-02-17 14:46:16 +00:00
|
|
|
dev->data.tcp.haveTLS != VIR_TRISTATE_BOOL_YES) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2017-02-22 17:54:10 +00:00
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
if (qemuDomainSecretChardevPrepare(cfg, priv, devAlias, dev) < 0)
|
2017-02-17 14:37:34 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2017-02-17 15:06:14 +00:00
|
|
|
if ((chrSourcePriv = QEMU_DOMAIN_CHR_SOURCE_PRIVATE(dev)))
|
|
|
|
secinfo = chrSourcePriv->secinfo;
|
|
|
|
|
2018-05-29 18:03:07 +00:00
|
|
|
if (secinfo)
|
|
|
|
*secAlias = secinfo->s.aes.alias;
|
|
|
|
|
2018-05-30 07:24:35 +00:00
|
|
|
if (!(*tlsAlias = qemuAliasTLSObjFromSrcAlias(charAlias)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2017-02-17 15:06:14 +00:00
|
|
|
if (qemuDomainGetTLSObjects(priv->qemuCaps, secinfo,
|
|
|
|
cfg->chardevTLSx509certdir,
|
|
|
|
dev->data.tcp.listen,
|
|
|
|
cfg->chardevTLSx509verify,
|
2018-05-22 05:38:22 +00:00
|
|
|
*tlsAlias, &tlsProps, &secProps) < 0)
|
2017-02-22 17:54:10 +00:00
|
|
|
goto cleanup;
|
2017-02-17 15:06:14 +00:00
|
|
|
dev->data.tcp.tlscreds = true;
|
2017-02-22 17:54:10 +00:00
|
|
|
|
2017-03-09 14:20:27 +00:00
|
|
|
if (qemuDomainAddTLSObjects(driver, vm, QEMU_ASYNC_JOB_NONE,
|
2018-05-22 05:38:22 +00:00
|
|
|
&secProps, &tlsProps) < 0)
|
2017-02-22 17:54:10 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(tlsProps);
|
|
|
|
virJSONValueFree(secProps);
|
2017-02-17 14:46:16 +00:00
|
|
|
virObjectUnref(cfg);
|
2017-02-22 17:54:10 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-19 22:46:41 +00:00
|
|
|
static int
|
|
|
|
qemuDomainDelChardevTLSObjects(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2017-12-20 11:36:26 +00:00
|
|
|
virDomainChrSourceDefPtr dev,
|
2017-12-19 22:46:41 +00:00
|
|
|
const char *inAlias)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
char *tlsAlias = NULL;
|
|
|
|
char *secAlias = NULL;
|
|
|
|
|
2017-12-20 11:36:26 +00:00
|
|
|
if (dev->type != VIR_DOMAIN_CHR_TYPE_TCP ||
|
|
|
|
dev->data.tcp.haveTLS != VIR_TRISTATE_BOOL_YES) {
|
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2017-12-19 22:46:41 +00:00
|
|
|
if (!(tlsAlias = qemuAliasTLSObjFromSrcAlias(inAlias)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* Best shot at this as the secinfo is destroyed after process launch
|
|
|
|
* and this path does not recreate it. Thus, if the config has the
|
|
|
|
* secret UUID and we have a serial TCP chardev, then formulate a
|
|
|
|
* secAlias which we'll attempt to destroy. */
|
|
|
|
if (cfg->chardevTLSx509secretUUID &&
|
|
|
|
!(secAlias = qemuDomainGetSecretAESAlias(inAlias, false)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
|
|
|
if (secAlias)
|
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, secAlias));
|
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(tlsAlias);
|
|
|
|
VIR_FREE(secAlias);
|
|
|
|
virObjectUnref(cfg);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
int qemuDomainAttachRedirdevDevice(virQEMUDriverPtr driver,
|
2011-09-02 15:09:14 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainRedirdevDefPtr redirdev)
|
|
|
|
{
|
2016-02-22 16:44:09 +00:00
|
|
|
int ret = -1;
|
2011-09-02 15:09:14 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2012-09-13 07:25:46 +00:00
|
|
|
virDomainDefPtr def = vm->def;
|
2016-02-22 16:44:09 +00:00
|
|
|
char *charAlias = NULL;
|
2011-09-02 15:09:14 +00:00
|
|
|
char *devstr = NULL;
|
2016-10-21 13:53:30 +00:00
|
|
|
bool chardevAdded = false;
|
2016-10-21 13:59:53 +00:00
|
|
|
char *tlsAlias = NULL;
|
2018-05-29 18:03:07 +00:00
|
|
|
const char *secAlias = NULL;
|
2017-01-27 14:26:13 +00:00
|
|
|
bool need_release = false;
|
2016-10-21 13:53:30 +00:00
|
|
|
virErrorPtr orig_err;
|
2011-09-02 15:09:14 +00:00
|
|
|
|
2016-06-19 12:46:40 +00:00
|
|
|
if (qemuAssignDeviceRedirdevAlias(def, redirdev, -1) < 0)
|
2016-02-22 16:44:09 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-10-18 14:37:23 +00:00
|
|
|
if (!(charAlias = qemuAliasChardevFromDevAlias(redirdev->info.alias)))
|
2016-02-22 16:44:09 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2017-05-30 11:22:17 +00:00
|
|
|
if ((virDomainUSBAddressEnsure(priv->usbaddrs, &redirdev->info)) < 0)
|
2017-01-27 14:26:13 +00:00
|
|
|
goto cleanup;
|
2017-05-30 11:22:17 +00:00
|
|
|
need_release = true;
|
2017-01-27 14:26:13 +00:00
|
|
|
|
2014-12-16 15:29:00 +00:00
|
|
|
if (!(devstr = qemuBuildRedirdevDevStr(def, redirdev, priv->qemuCaps)))
|
2016-02-22 16:44:09 +00:00
|
|
|
goto cleanup;
|
2014-12-16 15:29:00 +00:00
|
|
|
|
2016-06-19 12:46:40 +00:00
|
|
|
if (VIR_REALLOC_N(def->redirdevs, def->nredirdevs+1) < 0)
|
2016-02-22 16:44:09 +00:00
|
|
|
goto cleanup;
|
2011-09-02 15:09:14 +00:00
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
if (qemuDomainAddChardevTLSObjects(driver, vm, redirdev->source,
|
2017-02-17 14:37:34 +00:00
|
|
|
redirdev->info.alias, charAlias,
|
|
|
|
&tlsAlias, &secAlias) < 0)
|
2017-02-16 15:33:35 +00:00
|
|
|
goto audit;
|
2016-06-17 13:44:30 +00:00
|
|
|
|
2017-02-16 15:33:35 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2016-10-21 13:59:53 +00:00
|
|
|
|
2016-02-22 16:44:09 +00:00
|
|
|
if (qemuMonitorAttachCharDev(priv->mon,
|
|
|
|
charAlias,
|
2016-10-24 12:24:51 +00:00
|
|
|
redirdev->source) < 0)
|
2016-10-21 13:53:30 +00:00
|
|
|
goto exit_monitor;
|
|
|
|
chardevAdded = true;
|
2015-01-07 12:12:18 +00:00
|
|
|
|
2016-10-21 13:53:30 +00:00
|
|
|
if (qemuMonitorAddDevice(priv->mon, devstr) < 0)
|
|
|
|
goto exit_monitor;
|
2011-09-02 15:09:14 +00:00
|
|
|
|
2016-02-22 16:44:09 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto audit;
|
2011-09-02 15:09:14 +00:00
|
|
|
|
2016-06-19 12:46:40 +00:00
|
|
|
def->redirdevs[def->nredirdevs++] = redirdev;
|
2016-02-22 16:44:09 +00:00
|
|
|
ret = 0;
|
|
|
|
audit:
|
|
|
|
virDomainAuditRedirdev(vm, redirdev, "attach", ret == 0);
|
|
|
|
cleanup:
|
2017-01-27 14:26:13 +00:00
|
|
|
if (ret < 0 && need_release)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &redirdev->info, NULL);
|
2016-10-21 13:59:53 +00:00
|
|
|
VIR_FREE(tlsAlias);
|
2016-02-22 16:44:09 +00:00
|
|
|
VIR_FREE(charAlias);
|
2011-09-02 15:09:14 +00:00
|
|
|
VIR_FREE(devstr);
|
2016-02-22 16:44:09 +00:00
|
|
|
return ret;
|
2016-10-21 13:53:30 +00:00
|
|
|
|
|
|
|
exit_monitor:
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2016-10-21 13:53:30 +00:00
|
|
|
/* detach associated chardev on error */
|
|
|
|
if (chardevAdded)
|
|
|
|
ignore_value(qemuMonitorDetachCharDev(priv->mon, charAlias));
|
2017-02-22 17:39:17 +00:00
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2017-03-09 14:20:27 +00:00
|
|
|
qemuDomainDelTLSObjects(driver, vm, QEMU_ASYNC_JOB_NONE,
|
|
|
|
secAlias, tlsAlias);
|
2016-10-21 13:53:30 +00:00
|
|
|
goto audit;
|
2011-09-02 15:09:14 +00:00
|
|
|
}
|
|
|
|
|
2015-01-27 17:30:15 +00:00
|
|
|
static int
|
|
|
|
qemuDomainChrPreInsert(virDomainDefPtr vmdef,
|
|
|
|
virDomainChrDefPtr chr)
|
2013-03-12 14:59:25 +00:00
|
|
|
{
|
|
|
|
if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE &&
|
|
|
|
chr->targetType == VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_SERIAL) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("attaching serial console is not supported"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virDomainChrFind(vmdef, chr)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("chardev already exists"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-01-27 17:30:15 +00:00
|
|
|
if (virDomainChrPreAlloc(vmdef, chr) < 0)
|
2013-03-12 14:59:25 +00:00
|
|
|
return -1;
|
|
|
|
|
2017-05-29 10:58:34 +00:00
|
|
|
/* Due to historical reasons, the first console is an alias to the
|
|
|
|
* first serial device (if such exists). If this is the case, we need to
|
|
|
|
* create an object for the first console as well.
|
|
|
|
*/
|
2015-01-27 17:30:15 +00:00
|
|
|
if (vmdef->nserials == 0 && vmdef->nconsoles == 0 &&
|
|
|
|
chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_SERIAL) {
|
|
|
|
if (!vmdef->consoles && VIR_ALLOC(vmdef->consoles) < 0)
|
|
|
|
return -1;
|
|
|
|
|
2017-05-29 10:58:34 +00:00
|
|
|
/* We'll be dealing with serials[0] directly, so NULL is fine here. */
|
|
|
|
if (!(vmdef->consoles[0] = virDomainChrDefNew(NULL))) {
|
2015-01-27 17:30:15 +00:00
|
|
|
VIR_FREE(vmdef->consoles);
|
2013-03-12 14:59:25 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2015-01-27 17:30:15 +00:00
|
|
|
vmdef->nconsoles++;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemuDomainChrInsertPreAlloced(virDomainDefPtr vmdef,
|
|
|
|
virDomainChrDefPtr chr)
|
|
|
|
{
|
|
|
|
virDomainChrInsertPreAlloced(vmdef, chr);
|
|
|
|
if (vmdef->nserials == 1 && vmdef->nconsoles == 0 &&
|
|
|
|
chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_SERIAL) {
|
2013-03-12 14:59:25 +00:00
|
|
|
vmdef->nconsoles = 1;
|
|
|
|
|
|
|
|
/* Create an console alias for the serial port */
|
|
|
|
vmdef->consoles[0]->deviceType = VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE;
|
|
|
|
vmdef->consoles[0]->targetType = VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_SERIAL;
|
|
|
|
}
|
2015-01-27 17:30:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemuDomainChrInsertPreAllocCleanup(virDomainDefPtr vmdef,
|
|
|
|
virDomainChrDefPtr chr)
|
|
|
|
{
|
|
|
|
/* Remove the stub console added by qemuDomainChrPreInsert */
|
|
|
|
if (vmdef->nserials == 0 && vmdef->nconsoles == 1 &&
|
|
|
|
chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_SERIAL) {
|
2017-05-29 10:58:34 +00:00
|
|
|
virDomainChrDefFree(vmdef->consoles[0]);
|
2015-01-27 17:30:15 +00:00
|
|
|
VIR_FREE(vmdef->consoles);
|
|
|
|
vmdef->nconsoles = 0;
|
|
|
|
}
|
|
|
|
}
|
2013-03-12 14:59:25 +00:00
|
|
|
|
2015-01-27 17:30:15 +00:00
|
|
|
int
|
|
|
|
qemuDomainChrInsert(virDomainDefPtr vmdef,
|
|
|
|
virDomainChrDefPtr chr)
|
|
|
|
{
|
|
|
|
if (qemuDomainChrPreInsert(vmdef, chr) < 0) {
|
|
|
|
qemuDomainChrInsertPreAllocCleanup(vmdef, chr);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
qemuDomainChrInsertPreAlloced(vmdef, chr);
|
2013-03-12 14:59:25 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainChrDefPtr
|
|
|
|
qemuDomainChrRemove(virDomainDefPtr vmdef,
|
|
|
|
virDomainChrDefPtr chr)
|
|
|
|
{
|
|
|
|
virDomainChrDefPtr ret;
|
|
|
|
bool removeCompat;
|
|
|
|
|
|
|
|
if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE &&
|
|
|
|
chr->targetType == VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_SERIAL) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("detaching serial console is not supported"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Due to some crazy backcompat stuff, the first serial device is an alias
|
|
|
|
* to the first console too. If this is the case, the definition must be
|
|
|
|
* duplicated as first console device. */
|
|
|
|
removeCompat = vmdef->nserials && vmdef->nconsoles &&
|
|
|
|
vmdef->consoles[0]->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE &&
|
|
|
|
vmdef->consoles[0]->targetType == VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_SERIAL &&
|
|
|
|
virDomainChrEquals(vmdef->serials[0], chr);
|
|
|
|
|
|
|
|
if (!(ret = virDomainChrRemove(vmdef, chr))) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG, "%s",
|
|
|
|
_("device not present in domain configuration"));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (removeCompat)
|
|
|
|
VIR_DELETE_ELEMENT(vmdef->consoles, 0, vmdef->nconsoles);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2013-03-13 10:08:55 +00:00
|
|
|
|
2016-10-21 12:18:54 +00:00
|
|
|
/* Returns 1 if the address will need to be released later,
|
|
|
|
* -1 on error
|
|
|
|
* 0 otherwise
|
|
|
|
*/
|
2015-06-10 14:49:37 +00:00
|
|
|
static int
|
2016-10-12 19:24:57 +00:00
|
|
|
qemuDomainAttachChrDeviceAssignAddr(virDomainObjPtr vm,
|
2016-11-03 20:33:32 +00:00
|
|
|
virDomainChrDefPtr chr,
|
|
|
|
virQEMUDriverPtr driver)
|
2015-06-10 14:49:37 +00:00
|
|
|
{
|
2016-10-12 19:24:57 +00:00
|
|
|
virDomainDefPtr def = vm->def;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-09-07 16:29:30 +00:00
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_CHR, { .chr = chr } };
|
2016-07-23 01:47:07 +00:00
|
|
|
|
2015-06-10 14:49:37 +00:00
|
|
|
if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CONSOLE &&
|
|
|
|
chr->targetType == VIR_DOMAIN_CHR_CONSOLE_TARGET_TYPE_VIRTIO) {
|
2016-10-21 11:09:36 +00:00
|
|
|
if (virDomainVirtioSerialAddrAutoAssign(def, &chr->info, true) < 0)
|
2016-10-21 12:17:31 +00:00
|
|
|
return -1;
|
2016-10-21 12:18:54 +00:00
|
|
|
return 0;
|
2015-06-10 14:49:37 +00:00
|
|
|
|
|
|
|
} else if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_SERIAL &&
|
|
|
|
chr->targetType == VIR_DOMAIN_CHR_SERIAL_TARGET_TYPE_PCI) {
|
2016-11-03 20:33:32 +00:00
|
|
|
if (qemuDomainEnsurePCIAddress(vm, &dev, driver) < 0)
|
2016-10-21 12:17:31 +00:00
|
|
|
return -1;
|
|
|
|
return 1;
|
2015-06-10 14:49:37 +00:00
|
|
|
|
2017-10-20 11:28:21 +00:00
|
|
|
} else if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_SERIAL &&
|
2015-08-12 14:52:19 +00:00
|
|
|
chr->targetType == VIR_DOMAIN_CHR_SERIAL_TARGET_TYPE_USB) {
|
|
|
|
if (virDomainUSBAddressEnsure(priv->usbaddrs, &chr->info) < 0)
|
2016-10-21 12:17:31 +00:00
|
|
|
return -1;
|
|
|
|
return 1;
|
2015-08-12 14:52:19 +00:00
|
|
|
|
2015-06-10 14:49:37 +00:00
|
|
|
} else if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CHANNEL &&
|
|
|
|
chr->targetType == VIR_DOMAIN_CHR_CHANNEL_TARGET_TYPE_VIRTIO) {
|
2016-10-21 11:09:36 +00:00
|
|
|
if (virDomainVirtioSerialAddrAutoAssign(def, &chr->info, false) < 0)
|
2016-10-21 12:17:31 +00:00
|
|
|
return -1;
|
2016-10-21 12:18:54 +00:00
|
|
|
return 0;
|
2015-06-10 14:49:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (chr->info.type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_VIRTIO_SERIAL ||
|
|
|
|
chr->info.type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("Unsupported address type for character device"));
|
2016-10-21 12:17:31 +00:00
|
|
|
return -1;
|
2015-06-10 14:49:37 +00:00
|
|
|
}
|
|
|
|
|
2016-10-21 12:17:31 +00:00
|
|
|
return 0;
|
2015-06-10 14:49:37 +00:00
|
|
|
}
|
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
int qemuDomainAttachChrDevice(virQEMUDriverPtr driver,
|
2013-03-13 10:08:55 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainChrDefPtr chr)
|
|
|
|
{
|
2015-06-10 14:49:37 +00:00
|
|
|
int ret = -1, rc;
|
2013-03-13 10:08:55 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-07-14 21:55:05 +00:00
|
|
|
virErrorPtr orig_err;
|
2013-03-13 10:08:55 +00:00
|
|
|
virDomainDefPtr vmdef = vm->def;
|
|
|
|
char *devstr = NULL;
|
2016-10-21 11:45:54 +00:00
|
|
|
virDomainChrSourceDefPtr dev = chr->source;
|
2013-03-13 10:08:55 +00:00
|
|
|
char *charAlias = NULL;
|
2016-07-14 21:55:05 +00:00
|
|
|
bool chardevAttached = false;
|
2016-11-18 10:45:44 +00:00
|
|
|
bool teardowncgroup = false;
|
2016-11-18 13:53:27 +00:00
|
|
|
bool teardowndevice = false;
|
2017-12-01 12:10:35 +00:00
|
|
|
bool teardownlabel = false;
|
2016-06-13 16:30:34 +00:00
|
|
|
char *tlsAlias = NULL;
|
2018-05-29 18:03:07 +00:00
|
|
|
const char *secAlias = NULL;
|
2015-03-02 09:59:52 +00:00
|
|
|
bool need_release = false;
|
2019-02-11 15:05:37 +00:00
|
|
|
bool guestfwd = false;
|
2013-03-13 10:08:55 +00:00
|
|
|
|
2019-02-11 15:05:37 +00:00
|
|
|
if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CHANNEL) {
|
|
|
|
guestfwd = chr->targetType == VIR_DOMAIN_CHR_CHANNEL_TARGET_TYPE_GUESTFWD;
|
|
|
|
|
|
|
|
if (qemuDomainPrepareChannel(chr, priv->channelTargetDir) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2016-03-30 14:43:28 +00:00
|
|
|
|
2013-03-13 10:08:55 +00:00
|
|
|
if (qemuAssignDeviceChrAlias(vmdef, chr, -1) < 0)
|
2015-01-27 17:44:30 +00:00
|
|
|
goto cleanup;
|
2013-03-13 10:08:55 +00:00
|
|
|
|
2016-11-03 20:33:32 +00:00
|
|
|
if ((rc = qemuDomainAttachChrDeviceAssignAddr(vm, chr, driver)) < 0)
|
2015-06-10 14:49:37 +00:00
|
|
|
goto cleanup;
|
|
|
|
if (rc == 1)
|
|
|
|
need_release = true;
|
2015-03-02 09:59:52 +00:00
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceSetupChardev(vm, chr) < 0)
|
2016-11-18 13:53:27 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
2017-12-01 12:10:35 +00:00
|
|
|
if (qemuSecuritySetChardevLabel(driver, vm, chr) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardownlabel = true;
|
|
|
|
|
2016-11-18 10:45:44 +00:00
|
|
|
if (qemuSetupChardevCgroup(vm, chr) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
2016-06-19 12:46:40 +00:00
|
|
|
if (qemuBuildChrDeviceStr(&devstr, vmdef, chr, priv->qemuCaps) < 0)
|
2015-01-27 17:44:30 +00:00
|
|
|
goto cleanup;
|
2013-03-13 10:08:55 +00:00
|
|
|
|
2016-10-18 14:37:23 +00:00
|
|
|
if (!(charAlias = qemuAliasChardevFromDevAlias(chr->info.alias)))
|
2013-03-13 10:08:55 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2015-01-27 17:44:30 +00:00
|
|
|
if (qemuDomainChrPreInsert(vmdef, chr) < 0)
|
2013-03-13 10:08:55 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
if (qemuDomainAddChardevTLSObjects(driver, vm, dev,
|
2017-02-17 14:37:34 +00:00
|
|
|
chr->info.alias, charAlias,
|
2017-02-22 17:54:10 +00:00
|
|
|
&tlsAlias, &secAlias) < 0)
|
2017-02-16 15:33:35 +00:00
|
|
|
goto audit;
|
2016-06-17 13:44:30 +00:00
|
|
|
|
2017-02-16 15:33:35 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2016-06-13 16:30:34 +00:00
|
|
|
|
2016-10-21 11:45:54 +00:00
|
|
|
if (qemuMonitorAttachCharDev(priv->mon, charAlias, chr->source) < 0)
|
2016-07-14 21:55:05 +00:00
|
|
|
goto exit_monitor;
|
|
|
|
chardevAttached = true;
|
2016-06-13 16:57:15 +00:00
|
|
|
|
2019-02-11 15:05:37 +00:00
|
|
|
if (guestfwd) {
|
|
|
|
if (qemuMonitorAddNetdev(priv->mon, devstr,
|
|
|
|
NULL, NULL, 0, NULL, NULL, 0) < 0)
|
|
|
|
goto exit_monitor;
|
|
|
|
} else {
|
|
|
|
if (qemuMonitorAddDevice(priv->mon, devstr) < 0)
|
|
|
|
goto exit_monitor;
|
|
|
|
}
|
2013-03-13 10:08:55 +00:00
|
|
|
|
2015-01-27 17:44:30 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto audit;
|
2013-03-13 10:08:55 +00:00
|
|
|
|
2016-06-19 12:46:40 +00:00
|
|
|
qemuDomainChrInsertPreAlloced(vmdef, chr);
|
2013-03-13 10:08:55 +00:00
|
|
|
ret = 0;
|
2014-07-03 08:59:58 +00:00
|
|
|
audit:
|
|
|
|
virDomainAuditChardev(vm, NULL, chr, "attach", ret == 0);
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2016-11-18 10:45:44 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
if (virDomainObjIsActive(vm))
|
|
|
|
qemuDomainChrInsertPreAllocCleanup(vmdef, chr);
|
|
|
|
if (need_release)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &chr->info, NULL);
|
|
|
|
if (teardowncgroup && qemuTeardownChardevCgroup(vm, chr) < 0)
|
|
|
|
VIR_WARN("Unable to remove chr device cgroup ACL on hotplug fail");
|
2017-12-01 12:10:35 +00:00
|
|
|
if (teardownlabel && qemuSecurityRestoreChardevLabel(driver, vm, chr) < 0)
|
|
|
|
VIR_WARN("Unable to restore security label on char device");
|
2017-11-24 16:52:15 +00:00
|
|
|
if (teardowndevice && qemuDomainNamespaceTeardownChardev(vm, chr) < 0)
|
2016-11-18 13:53:27 +00:00
|
|
|
VIR_WARN("Unable to remove chr device from /dev");
|
2016-11-18 10:45:44 +00:00
|
|
|
}
|
2016-06-13 16:30:34 +00:00
|
|
|
VIR_FREE(tlsAlias);
|
2013-03-13 10:08:55 +00:00
|
|
|
VIR_FREE(charAlias);
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
return ret;
|
2016-06-13 16:57:15 +00:00
|
|
|
|
2016-07-14 21:55:05 +00:00
|
|
|
exit_monitor:
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2016-06-13 16:57:15 +00:00
|
|
|
/* detach associated chardev on error */
|
2016-07-14 21:55:05 +00:00
|
|
|
if (chardevAttached)
|
|
|
|
qemuMonitorDetachCharDev(priv->mon, charAlias);
|
2017-02-22 17:39:17 +00:00
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2016-07-14 21:55:05 +00:00
|
|
|
|
2017-03-09 14:20:27 +00:00
|
|
|
qemuDomainDelTLSObjects(driver, vm, QEMU_ASYNC_JOB_NONE,
|
|
|
|
secAlias, tlsAlias);
|
2016-06-13 16:57:15 +00:00
|
|
|
goto audit;
|
2013-03-13 10:08:55 +00:00
|
|
|
}
|
|
|
|
|
2015-01-17 05:09:37 +00:00
|
|
|
|
|
|
|
int
|
2018-02-09 16:14:41 +00:00
|
|
|
qemuDomainAttachRNGDevice(virQEMUDriverPtr driver,
|
2015-01-17 05:09:37 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainRNGDefPtr rng)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-09-07 16:29:30 +00:00
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_RNG, { .rng = rng } };
|
2016-07-14 22:13:50 +00:00
|
|
|
virErrorPtr orig_err;
|
2015-01-17 05:09:37 +00:00
|
|
|
char *devstr = NULL;
|
|
|
|
char *charAlias = NULL;
|
|
|
|
char *objAlias = NULL;
|
2016-10-21 14:06:50 +00:00
|
|
|
char *tlsAlias = NULL;
|
2018-05-29 18:03:07 +00:00
|
|
|
const char *secAlias = NULL;
|
2016-07-14 22:13:50 +00:00
|
|
|
bool releaseaddr = false;
|
2016-11-18 10:17:51 +00:00
|
|
|
bool teardowncgroup = false;
|
2016-11-18 14:19:12 +00:00
|
|
|
bool teardowndevice = false;
|
2016-07-14 22:13:50 +00:00
|
|
|
bool chardevAdded = false;
|
2015-01-17 05:09:37 +00:00
|
|
|
virJSONValuePtr props = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
2016-04-06 15:32:12 +00:00
|
|
|
if (qemuAssignDeviceRNGAlias(vm->def, rng) < 0)
|
2016-10-21 14:06:50 +00:00
|
|
|
goto cleanup;
|
2015-01-17 05:09:37 +00:00
|
|
|
|
|
|
|
/* preallocate space for the device definition */
|
|
|
|
if (VIR_REALLOC_N(vm->def->rngs, vm->def->nrngs + 1) < 0)
|
2016-10-21 14:06:50 +00:00
|
|
|
goto cleanup;
|
2015-01-17 05:09:37 +00:00
|
|
|
|
2017-10-12 12:53:27 +00:00
|
|
|
if (qemuDomainEnsureVirtioAddress(&releaseaddr, vm, &dev, "rng") < 0)
|
|
|
|
return -1;
|
2015-01-17 05:09:37 +00:00
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceSetupRNG(vm, rng) < 0)
|
2016-11-18 14:19:12 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
2016-11-18 10:17:51 +00:00
|
|
|
if (qemuSetupRNGCgroup(vm, rng) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
2015-01-17 05:09:37 +00:00
|
|
|
/* build required metadata */
|
|
|
|
if (!(devstr = qemuBuildRNGDevStr(vm->def, rng, priv->qemuCaps)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-18 12:48:22 +00:00
|
|
|
if (qemuBuildRNGBackendProps(rng, priv->qemuCaps, &props) < 0)
|
2015-01-17 05:09:37 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-10-18 14:37:23 +00:00
|
|
|
if (!(charAlias = qemuAliasChardevFromDevAlias(rng->info.alias)))
|
2015-01-17 05:09:37 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-06-17 13:44:30 +00:00
|
|
|
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD) {
|
2018-02-09 16:14:41 +00:00
|
|
|
if (qemuDomainAddChardevTLSObjects(driver, vm,
|
2017-02-17 14:37:34 +00:00
|
|
|
rng->source.chardev,
|
|
|
|
rng->info.alias, charAlias,
|
|
|
|
&tlsAlias, &secAlias) < 0)
|
2017-02-16 15:33:35 +00:00
|
|
|
goto audit;
|
2016-06-17 13:44:30 +00:00
|
|
|
}
|
|
|
|
|
2017-02-16 15:33:35 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2016-10-21 14:06:50 +00:00
|
|
|
|
2015-01-17 05:09:37 +00:00
|
|
|
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD &&
|
|
|
|
qemuMonitorAttachCharDev(priv->mon, charAlias,
|
|
|
|
rng->source.chardev) < 0)
|
2016-07-14 22:13:50 +00:00
|
|
|
goto exit_monitor;
|
|
|
|
chardevAdded = true;
|
2015-01-17 05:09:37 +00:00
|
|
|
|
2018-05-18 12:48:22 +00:00
|
|
|
if (qemuMonitorAddObject(priv->mon, &props, &objAlias) < 0)
|
2016-07-14 22:13:50 +00:00
|
|
|
goto exit_monitor;
|
2015-01-17 05:09:37 +00:00
|
|
|
|
2018-11-08 11:00:30 +00:00
|
|
|
if (qemuDomainAttachExtensionDevice(priv->mon, &rng->info) < 0)
|
2016-07-14 22:13:50 +00:00
|
|
|
goto exit_monitor;
|
2015-01-17 05:09:37 +00:00
|
|
|
|
2018-11-08 11:00:30 +00:00
|
|
|
if (qemuMonitorAddDevice(priv->mon, devstr) < 0) {
|
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, &rng->info));
|
|
|
|
goto exit_monitor;
|
|
|
|
}
|
|
|
|
|
2015-01-17 05:09:37 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
2016-07-14 22:13:50 +00:00
|
|
|
releaseaddr = false;
|
2015-01-17 05:09:37 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2016-05-17 10:59:43 +00:00
|
|
|
VIR_APPEND_ELEMENT_INPLACE(vm->def->rngs, vm->def->nrngs, rng);
|
2015-01-17 05:09:37 +00:00
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
audit:
|
|
|
|
virDomainAuditRNG(vm, NULL, rng, "attach", ret == 0);
|
|
|
|
cleanup:
|
2016-04-11 14:00:32 +00:00
|
|
|
virJSONValueFree(props);
|
2016-11-18 10:17:51 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
if (releaseaddr)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &rng->info, NULL);
|
|
|
|
if (teardowncgroup && qemuTeardownRNGCgroup(vm, rng) < 0)
|
|
|
|
VIR_WARN("Unable to remove RNG device cgroup ACL on hotplug fail");
|
2017-11-24 16:52:15 +00:00
|
|
|
if (teardowndevice && qemuDomainNamespaceTeardownRNG(vm, rng) < 0)
|
2016-11-18 14:19:12 +00:00
|
|
|
VIR_WARN("Unable to remove chr device from /dev");
|
2016-11-18 10:17:51 +00:00
|
|
|
}
|
|
|
|
|
2016-10-21 14:06:50 +00:00
|
|
|
VIR_FREE(tlsAlias);
|
2015-01-17 05:09:37 +00:00
|
|
|
VIR_FREE(charAlias);
|
|
|
|
VIR_FREE(objAlias);
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
return ret;
|
|
|
|
|
2016-07-14 22:13:50 +00:00
|
|
|
exit_monitor:
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2018-05-18 12:48:22 +00:00
|
|
|
if (objAlias)
|
2016-07-14 22:13:50 +00:00
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, objAlias));
|
|
|
|
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD && chardevAdded)
|
2015-01-17 05:09:37 +00:00
|
|
|
ignore_value(qemuMonitorDetachCharDev(priv->mon, charAlias));
|
2017-02-22 17:39:17 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
releaseaddr = false;
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2015-01-17 05:09:37 +00:00
|
|
|
|
2017-03-09 14:20:27 +00:00
|
|
|
qemuDomainDelTLSObjects(driver, vm, QEMU_ASYNC_JOB_NONE,
|
|
|
|
secAlias, tlsAlias);
|
2015-01-17 05:09:37 +00:00
|
|
|
goto audit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-12 22:28:58 +00:00
|
|
|
/**
|
|
|
|
* qemuDomainAttachMemory:
|
|
|
|
* @driver: qemu driver data
|
|
|
|
* @vm: VM object
|
|
|
|
* @mem: Definition of the memory device to be attached. @mem is always consumed
|
|
|
|
*
|
|
|
|
* Attaches memory device described by @mem to domain @vm.
|
|
|
|
*
|
|
|
|
* Returns 0 on success -1 on error.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
qemuDomainAttachMemory(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainMemoryDefPtr mem)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-07-15 11:27:48 +00:00
|
|
|
virErrorPtr orig_err;
|
2014-10-12 22:28:58 +00:00
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2016-06-15 13:34:04 +00:00
|
|
|
unsigned long long oldmem = virDomainDefGetMemoryTotal(vm->def);
|
2015-08-13 14:15:27 +00:00
|
|
|
unsigned long long newmem = oldmem + mem->size;
|
2014-10-12 22:28:58 +00:00
|
|
|
char *devstr = NULL;
|
|
|
|
char *objalias = NULL;
|
2016-07-15 11:27:48 +00:00
|
|
|
bool objAdded = false;
|
2016-08-04 13:26:09 +00:00
|
|
|
bool teardownlabel = false;
|
2017-02-22 15:33:12 +00:00
|
|
|
bool teardowncgroup = false;
|
2017-02-22 16:37:39 +00:00
|
|
|
bool teardowndevice = false;
|
2014-10-12 22:28:58 +00:00
|
|
|
virJSONValuePtr props = NULL;
|
2015-03-30 18:08:47 +00:00
|
|
|
virObjectEventPtr event;
|
2014-10-12 22:28:58 +00:00
|
|
|
int id;
|
|
|
|
int ret = -1;
|
|
|
|
|
2015-10-08 04:06:15 +00:00
|
|
|
qemuDomainMemoryDeviceAlignSize(vm->def, mem);
|
|
|
|
|
|
|
|
if (qemuDomainDefValidateMemoryHotplug(vm->def, priv->qemuCaps, mem) < 0)
|
2015-04-28 15:33:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-01 05:07:09 +00:00
|
|
|
if (qemuDomainAssignMemoryDeviceSlot(vm->def, mem) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-01 05:21:36 +00:00
|
|
|
/* in cases where we are using a VM with aliases generated according to the
|
|
|
|
* index of the memory device we need to keep continue using that scheme */
|
|
|
|
if (qemuAssignDeviceMemoryAlias(vm->def, mem, priv->memAliasOrderMismatch) < 0)
|
2014-10-12 22:28:58 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virAsprintf(&objalias, "mem%s", mem->info.alias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-12-20 09:14:49 +00:00
|
|
|
if (!(devstr = qemuBuildMemoryDeviceStr(mem, priv)))
|
2014-10-12 22:28:58 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-18 12:48:22 +00:00
|
|
|
if (qemuBuildMemoryBackendProps(&props, objalias, cfg,
|
2018-11-07 10:14:14 +00:00
|
|
|
priv, vm->def, mem, true) < 0)
|
2014-10-12 22:28:58 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2017-11-07 14:19:43 +00:00
|
|
|
if (qemuProcessBuildDestroyMemoryPaths(driver, vm, mem, true) < 0)
|
2017-06-07 12:47:37 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceSetupMemory(vm, mem) < 0)
|
2017-02-22 16:37:39 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
2017-02-22 15:33:12 +00:00
|
|
|
if (qemuSetupMemoryDevicesCgroup(vm, mem) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
2016-08-04 13:26:09 +00:00
|
|
|
if (qemuSecuritySetMemoryLabel(driver, vm, mem) < 0)
|
2014-10-12 22:28:58 +00:00
|
|
|
goto cleanup;
|
2016-08-04 13:26:09 +00:00
|
|
|
teardownlabel = true;
|
2014-10-12 22:28:58 +00:00
|
|
|
|
2016-08-04 13:26:09 +00:00
|
|
|
if (virDomainMemoryInsert(vm->def, mem) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuDomainAdjustMaxMemLock(vm) < 0)
|
2015-11-06 15:39:31 +00:00
|
|
|
goto removedef;
|
|
|
|
|
2014-10-12 22:28:58 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2018-05-18 12:48:22 +00:00
|
|
|
if (qemuMonitorAddObject(priv->mon, &props, NULL) < 0)
|
2015-11-06 15:39:31 +00:00
|
|
|
goto exit_monitor;
|
2016-07-15 11:27:48 +00:00
|
|
|
objAdded = true;
|
2014-10-12 22:28:58 +00:00
|
|
|
|
2016-07-15 11:27:48 +00:00
|
|
|
if (qemuMonitorAddDevice(priv->mon, devstr) < 0)
|
2015-11-06 15:39:31 +00:00
|
|
|
goto exit_monitor;
|
2014-10-12 22:28:58 +00:00
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
/* we shouldn't touch mem now, as the def might be freed */
|
|
|
|
mem = NULL;
|
2015-08-13 14:15:27 +00:00
|
|
|
goto audit;
|
2014-10-12 22:28:58 +00:00
|
|
|
}
|
|
|
|
|
2015-03-30 18:08:47 +00:00
|
|
|
event = virDomainEventDeviceAddedNewFromObj(vm, objalias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2015-03-30 18:08:47 +00:00
|
|
|
|
2016-04-06 13:57:57 +00:00
|
|
|
/* fix the balloon size */
|
|
|
|
ignore_value(qemuProcessRefreshBalloonState(driver, vm, QEMU_ASYNC_JOB_NONE));
|
2015-04-30 16:03:41 +00:00
|
|
|
|
2014-10-12 22:28:58 +00:00
|
|
|
/* mem is consumed by vm->def */
|
|
|
|
mem = NULL;
|
|
|
|
|
|
|
|
/* this step is best effort, removing the device would be so much trouble */
|
|
|
|
ignore_value(qemuDomainUpdateMemoryDeviceInfo(driver, vm,
|
|
|
|
QEMU_ASYNC_JOB_NONE));
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2015-08-13 14:15:27 +00:00
|
|
|
audit:
|
|
|
|
virDomainAuditMemory(vm, oldmem, newmem, "update", ret == 0);
|
2014-10-12 22:28:58 +00:00
|
|
|
cleanup:
|
2016-08-04 13:26:09 +00:00
|
|
|
if (mem && ret < 0) {
|
2017-02-22 15:33:12 +00:00
|
|
|
if (teardowncgroup && qemuTeardownMemoryDevicesCgroup(vm, mem) < 0)
|
|
|
|
VIR_WARN("Unable to remove memory device cgroup ACL on hotplug fail");
|
2016-08-04 13:26:09 +00:00
|
|
|
if (teardownlabel && qemuSecurityRestoreMemoryLabel(driver, vm, mem) < 0)
|
|
|
|
VIR_WARN("Unable to restore security label on memdev");
|
2017-02-22 16:37:39 +00:00
|
|
|
if (teardowndevice &&
|
2017-11-24 16:52:15 +00:00
|
|
|
qemuDomainNamespaceTeardownMemory(vm, mem) < 0)
|
2017-02-22 16:37:39 +00:00
|
|
|
VIR_WARN("Unable to remove memory device from /dev");
|
2016-08-04 13:26:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virJSONValueFree(props);
|
2014-10-12 22:28:58 +00:00
|
|
|
virObjectUnref(cfg);
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
VIR_FREE(objalias);
|
|
|
|
virDomainMemoryDefFree(mem);
|
|
|
|
return ret;
|
|
|
|
|
2015-11-06 15:39:31 +00:00
|
|
|
exit_monitor:
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2016-07-15 11:27:48 +00:00
|
|
|
if (objAdded)
|
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, objalias));
|
2017-02-22 17:39:17 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
mem = NULL;
|
2018-03-13 17:19:39 +00:00
|
|
|
|
|
|
|
if (objAdded && mem)
|
|
|
|
ignore_value(qemuProcessDestroyMemoryBackingPath(driver, vm, mem));
|
|
|
|
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2017-02-22 17:39:17 +00:00
|
|
|
if (!mem)
|
2015-08-13 14:15:27 +00:00
|
|
|
goto audit;
|
2014-10-12 22:28:58 +00:00
|
|
|
|
2015-11-06 15:39:31 +00:00
|
|
|
removedef:
|
2014-10-12 22:28:58 +00:00
|
|
|
if ((id = virDomainMemoryFindByDef(vm->def, mem)) >= 0)
|
|
|
|
mem = virDomainMemoryRemove(vm->def, id);
|
|
|
|
else
|
|
|
|
mem = NULL;
|
|
|
|
|
2015-11-06 15:39:31 +00:00
|
|
|
/* reset the mlock limit */
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2015-11-23 16:57:40 +00:00
|
|
|
ignore_value(qemuDomainAdjustMaxMemLock(vm));
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2015-11-06 15:39:31 +00:00
|
|
|
|
2015-08-13 14:15:27 +00:00
|
|
|
goto audit;
|
2014-10-12 22:28:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-05 18:09:04 +00:00
|
|
|
static int
|
2014-03-06 07:35:03 +00:00
|
|
|
qemuDomainAttachHostUSBDevice(virQEMUDriverPtr driver,
|
2013-12-05 18:09:04 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
char *devstr = NULL;
|
2013-05-03 18:07:30 +00:00
|
|
|
bool added = false;
|
2013-11-14 11:02:40 +00:00
|
|
|
bool teardowncgroup = false;
|
2013-12-05 19:54:41 +00:00
|
|
|
bool teardownlabel = false;
|
2016-11-16 14:27:47 +00:00
|
|
|
bool teardowndevice = false;
|
2013-05-03 18:07:30 +00:00
|
|
|
int ret = -1;
|
|
|
|
|
2017-10-20 11:28:21 +00:00
|
|
|
if (virDomainUSBAddressEnsure(priv->usbaddrs, hostdev->info) < 0)
|
|
|
|
return -1;
|
2015-08-12 14:52:19 +00:00
|
|
|
|
2015-10-20 12:10:16 +00:00
|
|
|
if (qemuHostdevPrepareUSBDevices(driver, vm->def->name, &hostdev, 1, 0) < 0)
|
2013-05-03 18:07:30 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
added = true;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceSetupHostdev(vm, hostdev) < 0)
|
2016-11-16 14:27:47 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
2015-11-19 13:35:46 +00:00
|
|
|
if (qemuSetupHostdevCgroup(vm, hostdev) < 0)
|
2013-11-14 11:02:40 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
2016-11-16 14:27:47 +00:00
|
|
|
if (qemuSecuritySetHostdevLabel(driver, vm, hostdev) < 0)
|
2013-12-05 19:54:41 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardownlabel = true;
|
|
|
|
|
2016-04-26 11:40:34 +00:00
|
|
|
if (qemuAssignDeviceHostdevAlias(vm->def, &hostdev->info->alias, -1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
if (!(devstr = qemuBuildUSBHostdevDevStr(vm->def, hostdev, priv->qemuCaps)))
|
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-07-04 10:14:12 +00:00
|
|
|
if (VIR_REALLOC_N(vm->def->hostdevs, vm->def->nhostdevs+1) < 0)
|
2013-05-03 18:07:30 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-02-06 18:17:20 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2016-04-26 11:40:34 +00:00
|
|
|
ret = qemuMonitorAddDevice(priv->mon, devstr);
|
2015-01-07 12:12:18 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
ret = -1;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
Move qemu_audit.h helpers into shared code
The LXC and UML drivers can both make use of auditing. Move
the qemu_audit.{c,h} files to src/conf/domain_audit.{c,h}
* src/conf/domain_audit.c: Rename from src/qemu/qemu_audit.c
* src/conf/domain_audit.h: Rename from src/qemu/qemu_audit.h
* src/Makefile.am: Remove qemu_audit.{c,h}, add domain_audit.{c,h}
* src/qemu/qemu_audit.h, src/qemu/qemu_cgroup.c,
src/qemu/qemu_command.c, src/qemu/qemu_driver.c,
src/qemu/qemu_hotplug.c, src/qemu/qemu_migration.c,
src/qemu/qemu_process.c: Update for changed audit API names
2011-07-04 10:56:13 +00:00
|
|
|
virDomainAuditHostdev(vm, hostdev, "attach", ret == 0);
|
2010-12-16 16:10:54 +00:00
|
|
|
if (ret < 0)
|
2013-05-03 18:07:30 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
vm->def->hostdevs[vm->def->nhostdevs++] = hostdev;
|
|
|
|
|
2013-05-03 18:07:30 +00:00
|
|
|
ret = 0;
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2013-12-05 19:54:41 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
if (teardowncgroup && qemuTeardownHostdevCgroup(vm, hostdev) < 0)
|
|
|
|
VIR_WARN("Unable to remove host device cgroup ACL on hotplug fail");
|
|
|
|
if (teardownlabel &&
|
2016-11-16 14:27:47 +00:00
|
|
|
qemuSecurityRestoreHostdevLabel(driver, vm, hostdev) < 0)
|
2013-12-05 19:54:41 +00:00
|
|
|
VIR_WARN("Unable to restore host device labelling on hotplug fail");
|
2016-11-16 14:27:47 +00:00
|
|
|
if (teardowndevice &&
|
2017-11-24 16:52:15 +00:00
|
|
|
qemuDomainNamespaceTeardownHostdev(vm, hostdev) < 0)
|
2016-11-16 14:27:47 +00:00
|
|
|
VIR_WARN("Unable to remove host device from /dev");
|
2013-12-05 19:59:05 +00:00
|
|
|
if (added)
|
2015-10-20 12:12:48 +00:00
|
|
|
qemuHostdevReAttachUSBDevices(driver, vm->def->name, &hostdev, 1);
|
2017-10-20 11:24:49 +00:00
|
|
|
virDomainUSBAddressRelease(priv->usbaddrs, hostdev->info);
|
2013-12-05 19:54:41 +00:00
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
VIR_FREE(devstr);
|
2013-05-03 18:07:30 +00:00
|
|
|
return ret;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2016-04-06 14:41:33 +00:00
|
|
|
|
2013-05-03 18:07:31 +00:00
|
|
|
static int
|
2018-02-09 16:14:41 +00:00
|
|
|
qemuDomainAttachHostSCSIDevice(virQEMUDriverPtr driver,
|
2013-05-03 18:07:31 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
2016-06-27 14:43:47 +00:00
|
|
|
size_t i;
|
2013-05-03 18:07:31 +00:00
|
|
|
int ret = -1;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-04-11 17:32:12 +00:00
|
|
|
virErrorPtr orig_err;
|
2013-05-03 18:07:31 +00:00
|
|
|
char *devstr = NULL;
|
|
|
|
char *drvstr = NULL;
|
2016-07-18 17:24:27 +00:00
|
|
|
char *drivealias = NULL;
|
2018-05-22 05:38:22 +00:00
|
|
|
char *secobjAlias = NULL;
|
2013-11-14 11:02:40 +00:00
|
|
|
bool teardowncgroup = false;
|
2013-12-05 19:54:41 +00:00
|
|
|
bool teardownlabel = false;
|
2016-11-16 14:27:47 +00:00
|
|
|
bool teardowndevice = false;
|
2016-07-14 21:15:10 +00:00
|
|
|
bool driveAdded = false;
|
2017-09-15 17:17:59 +00:00
|
|
|
virJSONValuePtr secobjProps = NULL;
|
|
|
|
virDomainHostdevSubsysSCSIPtr scsisrc = &hostdev->source.subsys.u.scsi;
|
2017-12-05 21:03:34 +00:00
|
|
|
qemuDomainSecretInfoPtr secinfo = NULL;
|
2013-05-03 18:07:31 +00:00
|
|
|
|
2016-06-27 14:43:47 +00:00
|
|
|
/* Let's make sure the disk has a controller defined and loaded before
|
|
|
|
* trying to add it. The controller used by the disk must exist before a
|
|
|
|
* qemu command line string is generated.
|
|
|
|
*
|
|
|
|
* Ensure that the given controller and all controllers with a smaller index
|
|
|
|
* exist; there must not be any missing index in between.
|
|
|
|
*/
|
|
|
|
for (i = 0; i <= hostdev->info->addr.drive.controller; i++) {
|
|
|
|
if (!qemuDomainFindOrCreateSCSIDiskController(driver, vm, i))
|
|
|
|
return -1;
|
|
|
|
}
|
2013-11-21 03:36:27 +00:00
|
|
|
|
2017-04-26 21:10:00 +00:00
|
|
|
if (qemuHostdevPrepareSCSIDevices(driver, vm->def->name, &hostdev, 1) < 0)
|
2013-05-03 18:07:31 +00:00
|
|
|
return -1;
|
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceSetupHostdev(vm, hostdev) < 0)
|
2016-11-16 14:27:47 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
2015-11-19 13:35:46 +00:00
|
|
|
if (qemuSetupHostdevCgroup(vm, hostdev) < 0)
|
2013-11-14 11:02:40 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
2016-11-16 14:27:47 +00:00
|
|
|
if (qemuSecuritySetHostdevLabel(driver, vm, hostdev) < 0)
|
2013-12-05 19:54:41 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardownlabel = true;
|
|
|
|
|
2016-04-01 14:40:23 +00:00
|
|
|
if (qemuAssignDeviceHostdevAlias(vm->def, &hostdev->info->alias, -1) < 0)
|
2013-05-03 18:07:31 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-02-09 16:14:41 +00:00
|
|
|
if (qemuDomainSecretHostdevPrepare(priv, hostdev) < 0)
|
2016-04-06 14:41:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-01-17 12:26:08 +00:00
|
|
|
if (scsisrc->protocol == VIR_DOMAIN_HOSTDEV_SCSI_PROTOCOL_TYPE_ISCSI) {
|
|
|
|
qemuDomainStorageSourcePrivatePtr srcPriv =
|
|
|
|
QEMU_DOMAIN_STORAGE_SOURCE_PRIVATE(scsisrc->u.iscsi.src);
|
2018-07-03 08:45:34 +00:00
|
|
|
if (srcPriv)
|
|
|
|
secinfo = srcPriv->secinfo;
|
2018-01-17 12:26:08 +00:00
|
|
|
}
|
|
|
|
|
2017-09-15 17:17:59 +00:00
|
|
|
if (secinfo && secinfo->type == VIR_DOMAIN_SECRET_INFO_TYPE_AES) {
|
|
|
|
if (qemuBuildSecretInfoProps(secinfo, &secobjProps) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(drvstr = qemuBuildSCSIHostdevDrvStr(hostdev, priv->qemuCaps)))
|
2013-05-03 18:07:31 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-07-18 17:24:27 +00:00
|
|
|
if (!(drivealias = qemuAliasFromHostdev(hostdev)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-01-30 22:11:54 +00:00
|
|
|
if (!(devstr = qemuBuildSCSIHostdevDevStr(vm->def, hostdev)))
|
2013-05-03 18:07:31 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2013-07-04 10:14:12 +00:00
|
|
|
if (VIR_REALLOC_N(vm->def->hostdevs, vm->def->nhostdevs + 1) < 0)
|
2013-05-03 18:07:31 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
2018-05-22 05:38:22 +00:00
|
|
|
if (secobjProps &&
|
|
|
|
qemuMonitorAddObject(priv->mon, &secobjProps, &secobjAlias) < 0)
|
|
|
|
goto exit_monitor;
|
2017-09-15 17:17:59 +00:00
|
|
|
|
2016-04-11 17:32:12 +00:00
|
|
|
if (qemuMonitorAddDrive(priv->mon, drvstr) < 0)
|
2016-07-14 21:15:10 +00:00
|
|
|
goto exit_monitor;
|
|
|
|
driveAdded = true;
|
2016-04-11 17:32:12 +00:00
|
|
|
|
|
|
|
if (qemuMonitorAddDevice(priv->mon, devstr) < 0)
|
2016-07-14 21:15:10 +00:00
|
|
|
goto exit_monitor;
|
2016-04-11 17:32:12 +00:00
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
2016-07-14 21:15:10 +00:00
|
|
|
goto cleanup;
|
2016-04-11 17:32:12 +00:00
|
|
|
|
|
|
|
virDomainAuditHostdev(vm, hostdev, "attach", true);
|
2013-05-03 18:07:31 +00:00
|
|
|
|
|
|
|
vm->def->hostdevs[vm->def->nhostdevs++] = hostdev;
|
|
|
|
|
|
|
|
ret = 0;
|
2016-04-11 17:32:12 +00:00
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2013-11-14 11:02:40 +00:00
|
|
|
if (ret < 0) {
|
2015-10-20 12:12:48 +00:00
|
|
|
qemuHostdevReAttachSCSIDevices(driver, vm->def->name, &hostdev, 1);
|
2013-11-14 11:02:40 +00:00
|
|
|
if (teardowncgroup && qemuTeardownHostdevCgroup(vm, hostdev) < 0)
|
|
|
|
VIR_WARN("Unable to remove host device cgroup ACL on hotplug fail");
|
2013-12-05 19:54:41 +00:00
|
|
|
if (teardownlabel &&
|
2016-11-16 14:27:47 +00:00
|
|
|
qemuSecurityRestoreHostdevLabel(driver, vm, hostdev) < 0)
|
2013-12-05 19:54:41 +00:00
|
|
|
VIR_WARN("Unable to restore host device labelling on hotplug fail");
|
2016-11-16 14:27:47 +00:00
|
|
|
if (teardowndevice &&
|
2017-11-24 16:52:15 +00:00
|
|
|
qemuDomainNamespaceTeardownHostdev(vm, hostdev) < 0)
|
2016-11-16 14:27:47 +00:00
|
|
|
VIR_WARN("Unable to remove host device from /dev");
|
2013-11-14 11:02:40 +00:00
|
|
|
}
|
2017-09-15 17:17:59 +00:00
|
|
|
qemuDomainSecretHostdevDestroy(hostdev);
|
|
|
|
virJSONValueFree(secobjProps);
|
2018-05-22 05:38:22 +00:00
|
|
|
VIR_FREE(secobjAlias);
|
2016-07-18 17:24:27 +00:00
|
|
|
VIR_FREE(drivealias);
|
2013-05-03 18:07:31 +00:00
|
|
|
VIR_FREE(drvstr);
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
return ret;
|
2016-04-11 17:32:12 +00:00
|
|
|
|
2016-07-14 21:15:10 +00:00
|
|
|
exit_monitor:
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2016-07-18 17:24:27 +00:00
|
|
|
if (driveAdded && qemuMonitorDriveDel(priv->mon, drivealias) < 0) {
|
2016-04-11 17:32:12 +00:00
|
|
|
VIR_WARN("Unable to remove drive %s (%s) after failed "
|
|
|
|
"qemuMonitorAddDevice",
|
|
|
|
drvstr, devstr);
|
2016-07-14 21:15:10 +00:00
|
|
|
}
|
2018-05-22 05:38:22 +00:00
|
|
|
if (secobjAlias)
|
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, secobjAlias));
|
2017-02-22 17:39:17 +00:00
|
|
|
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2016-04-11 17:32:12 +00:00
|
|
|
|
|
|
|
virDomainAuditHostdev(vm, hostdev, "attach", false);
|
|
|
|
|
|
|
|
goto cleanup;
|
2013-05-03 18:07:31 +00:00
|
|
|
}
|
|
|
|
|
2016-11-22 03:58:19 +00:00
|
|
|
static int
|
|
|
|
qemuDomainAttachSCSIVHostDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_HOSTDEV,
|
|
|
|
{ .hostdev = hostdev } };
|
|
|
|
virDomainCCWAddressSetPtr ccwaddrs = NULL;
|
|
|
|
char *vhostfdName = NULL;
|
|
|
|
int vhostfd = -1;
|
|
|
|
char *devstr = NULL;
|
|
|
|
bool teardowncgroup = false;
|
|
|
|
bool teardownlabel = false;
|
2017-02-04 17:54:10 +00:00
|
|
|
bool teardowndevice = false;
|
2016-11-22 03:58:19 +00:00
|
|
|
bool releaseaddr = false;
|
|
|
|
|
2017-04-26 21:10:01 +00:00
|
|
|
if (qemuHostdevPrepareSCSIVHostDevices(driver, vm->def->name, &hostdev, 1) < 0)
|
2016-11-22 03:58:19 +00:00
|
|
|
return -1;
|
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceSetupHostdev(vm, hostdev) < 0)
|
2017-02-04 17:54:10 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
2016-11-22 03:58:19 +00:00
|
|
|
if (qemuSetupHostdevCgroup(vm, hostdev) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
2017-02-07 14:56:23 +00:00
|
|
|
if (qemuSecuritySetHostdevLabel(driver, vm, hostdev) < 0)
|
2016-11-22 03:58:19 +00:00
|
|
|
goto cleanup;
|
|
|
|
teardownlabel = true;
|
|
|
|
|
|
|
|
if (virSCSIVHostOpenVhostSCSI(&vhostfd) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virAsprintf(&vhostfdName, "vhostfd-%d", vhostfd) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (hostdev->info->type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE) {
|
2017-04-18 10:43:58 +00:00
|
|
|
if (qemuDomainIsS390CCW(vm->def) &&
|
2018-05-07 14:41:11 +00:00
|
|
|
virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_CCW))
|
2016-11-22 03:58:19 +00:00
|
|
|
hostdev->info->type = VIR_DOMAIN_DEVICE_ADDRESS_TYPE_CCW;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hostdev->info->type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE ||
|
|
|
|
hostdev->info->type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI) {
|
2016-11-03 20:33:32 +00:00
|
|
|
if (qemuDomainEnsurePCIAddress(vm, &dev, driver) < 0)
|
2016-11-22 03:58:19 +00:00
|
|
|
goto cleanup;
|
|
|
|
} else if (hostdev->info->type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_CCW) {
|
2018-07-03 15:25:28 +00:00
|
|
|
if (!(ccwaddrs = virDomainCCWAddressSetCreateFromDomain(vm->def)))
|
2016-11-22 03:58:19 +00:00
|
|
|
goto cleanup;
|
|
|
|
if (virDomainCCWAddressAssign(hostdev->info, ccwaddrs,
|
|
|
|
!hostdev->info->addr.ccw.assigned) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
releaseaddr = true;
|
|
|
|
|
|
|
|
if (qemuAssignDeviceHostdevAlias(vm->def, &hostdev->info->alias, -1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!(devstr = qemuBuildSCSIVHostHostdevDevStr(vm->def,
|
|
|
|
hostdev,
|
|
|
|
priv->qemuCaps,
|
|
|
|
vhostfdName)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (VIR_REALLOC_N(vm->def->hostdevs, vm->def->nhostdevs + 1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
2018-11-08 11:00:30 +00:00
|
|
|
if ((ret = qemuDomainAttachExtensionDevice(priv->mon, hostdev->info)) < 0)
|
|
|
|
goto exit_monitor;
|
|
|
|
|
|
|
|
if ((ret = qemuMonitorAddDeviceWithFd(priv->mon, devstr, vhostfd,
|
|
|
|
vhostfdName)) < 0) {
|
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, hostdev->info));
|
|
|
|
goto exit_monitor;
|
|
|
|
}
|
2016-11-22 03:58:19 +00:00
|
|
|
|
2018-11-08 11:00:30 +00:00
|
|
|
exit_monitor:
|
2016-11-22 03:58:19 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0 || ret < 0)
|
|
|
|
goto audit;
|
|
|
|
|
|
|
|
vm->def->hostdevs[vm->def->nhostdevs++] = hostdev;
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
audit:
|
|
|
|
virDomainAuditHostdev(vm, hostdev, "attach", (ret == 0));
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (ret < 0) {
|
|
|
|
if (teardowncgroup && qemuTeardownHostdevCgroup(vm, hostdev) < 0)
|
|
|
|
VIR_WARN("Unable to remove host device cgroup ACL on hotplug fail");
|
|
|
|
if (teardownlabel &&
|
2017-02-07 14:56:23 +00:00
|
|
|
qemuSecurityRestoreHostdevLabel(driver, vm, hostdev) < 0)
|
2016-11-22 03:58:19 +00:00
|
|
|
VIR_WARN("Unable to restore host device labelling on hotplug fail");
|
2017-02-04 17:54:10 +00:00
|
|
|
if (teardowndevice &&
|
2017-11-24 16:52:15 +00:00
|
|
|
qemuDomainNamespaceTeardownHostdev(vm, hostdev) < 0)
|
2017-02-04 17:54:10 +00:00
|
|
|
VIR_WARN("Unable to remove host device from /dev");
|
2016-11-22 03:58:19 +00:00
|
|
|
if (releaseaddr)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, hostdev->info, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainCCWAddressSetFree(ccwaddrs);
|
|
|
|
|
|
|
|
VIR_FORCE_CLOSE(vhostfd);
|
|
|
|
VIR_FREE(vhostfdName);
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-04-06 14:41:33 +00:00
|
|
|
|
2018-03-26 07:38:47 +00:00
|
|
|
static int
|
|
|
|
qemuDomainAttachMediatedDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
char *devstr = NULL;
|
|
|
|
bool added = false;
|
|
|
|
bool teardowncgroup = false;
|
|
|
|
bool teardownlabel = false;
|
|
|
|
bool teardowndevice = false;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_HOSTDEV,
|
|
|
|
{ .hostdev = hostdev } };
|
|
|
|
|
2018-06-26 11:47:39 +00:00
|
|
|
switch (hostdev->source.subsys.u.mdev.model) {
|
|
|
|
case VIR_MDEV_MODEL_TYPE_VFIO_PCI:
|
|
|
|
if (qemuDomainEnsurePCIAddress(vm, &dev, driver) < 0)
|
|
|
|
return -1;
|
|
|
|
break;
|
|
|
|
case VIR_MDEV_MODEL_TYPE_VFIO_CCW:
|
|
|
|
case VIR_MDEV_MODEL_TYPE_LAST:
|
|
|
|
break;
|
|
|
|
}
|
2018-03-26 07:38:47 +00:00
|
|
|
|
|
|
|
if (qemuHostdevPrepareMediatedDevices(driver,
|
|
|
|
vm->def->name,
|
|
|
|
&hostdev,
|
|
|
|
1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
added = true;
|
|
|
|
|
|
|
|
if (qemuDomainNamespaceSetupHostdev(vm, hostdev) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
|
|
|
if (qemuSetupHostdevCgroup(vm, hostdev) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
|
|
|
if (qemuSecuritySetHostdevLabel(driver, vm, hostdev) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardownlabel = true;
|
|
|
|
|
|
|
|
if (qemuAssignDeviceHostdevAlias(vm->def, &hostdev->info->alias, -1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!(devstr = qemuBuildHostdevMediatedDevStr(vm->def, hostdev,
|
|
|
|
priv->qemuCaps)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (VIR_REALLOC_N(vm->def->hostdevs, vm->def->nhostdevs + 1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
ret = qemuMonitorAddDevice(priv->mon, devstr);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
ret = -1;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainAuditHostdev(vm, hostdev, "attach", ret == 0);
|
|
|
|
if (ret < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
VIR_APPEND_ELEMENT_INPLACE(vm->def->hostdevs, vm->def->nhostdevs, hostdev);
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
if (ret < 0) {
|
|
|
|
if (teardowncgroup && qemuTeardownHostdevCgroup(vm, hostdev) < 0)
|
|
|
|
VIR_WARN("Unable to remove host device cgroup ACL on hotplug fail");
|
|
|
|
if (teardownlabel &&
|
|
|
|
qemuSecurityRestoreHostdevLabel(driver, vm, hostdev) < 0)
|
|
|
|
VIR_WARN("Unable to restore host device labelling on hotplug fail");
|
|
|
|
if (teardowndevice &&
|
|
|
|
qemuDomainNamespaceTeardownHostdev(vm, hostdev) < 0)
|
|
|
|
VIR_WARN("Unable to remove host device from /dev");
|
|
|
|
if (added)
|
|
|
|
qemuHostdevReAttachMediatedDevices(driver,
|
|
|
|
vm->def->name,
|
|
|
|
&hostdev,
|
|
|
|
1);
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, hostdev->info, NULL);
|
|
|
|
}
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-06 14:41:33 +00:00
|
|
|
int
|
2018-02-09 16:14:41 +00:00
|
|
|
qemuDomainAttachHostDevice(virQEMUDriverPtr driver,
|
2016-04-06 14:41:33 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
|
|
|
if (hostdev->mode != VIR_DOMAIN_HOSTDEV_MODE_SUBSYS) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
2017-05-11 13:26:47 +00:00
|
|
|
_("hotplug is not supported for hostdev mode '%s'"),
|
2012-07-18 15:22:03 +00:00
|
|
|
virDomainHostdevModeTypeToString(hostdev->mode));
|
2010-12-16 16:10:54 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (hostdev->source.subsys.type) {
|
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI:
|
2014-03-06 07:35:03 +00:00
|
|
|
if (qemuDomainAttachHostPCIDevice(driver, vm,
|
2011-05-04 12:09:09 +00:00
|
|
|
hostdev) < 0)
|
2010-12-16 16:10:54 +00:00
|
|
|
goto error;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB:
|
2014-03-06 07:35:03 +00:00
|
|
|
if (qemuDomainAttachHostUSBDevice(driver, vm,
|
2011-05-04 12:09:09 +00:00
|
|
|
hostdev) < 0)
|
2010-12-16 16:10:54 +00:00
|
|
|
goto error;
|
|
|
|
break;
|
|
|
|
|
2013-05-03 18:07:31 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI:
|
2018-02-09 16:14:41 +00:00
|
|
|
if (qemuDomainAttachHostSCSIDevice(driver, vm,
|
2013-05-03 18:07:31 +00:00
|
|
|
hostdev) < 0)
|
|
|
|
goto error;
|
|
|
|
break;
|
|
|
|
|
2016-11-22 03:58:19 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI_HOST:
|
|
|
|
if (qemuDomainAttachSCSIVHostDevice(driver, vm, hostdev) < 0)
|
|
|
|
goto error;
|
|
|
|
break;
|
2018-03-26 07:38:47 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_MDEV:
|
|
|
|
if (qemuDomainAttachMediatedDevice(driver, vm, hostdev) < 0)
|
|
|
|
goto error;
|
|
|
|
break;
|
2016-11-22 03:58:19 +00:00
|
|
|
|
2010-12-16 16:10:54 +00:00
|
|
|
default:
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
2017-05-11 13:26:47 +00:00
|
|
|
_("hotplug is not supported for hostdev subsys type '%s'"),
|
2012-07-18 15:22:03 +00:00
|
|
|
virDomainHostdevSubsysTypeToString(hostdev->source.subsys.type));
|
2010-12-16 16:10:54 +00:00
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
error:
|
2010-12-16 16:10:54 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-09-12 13:40:48 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainAttachShmemDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainShmemDefPtr shmem)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
char *shmstr = NULL;
|
|
|
|
char *charAlias = NULL;
|
|
|
|
char *memAlias = NULL;
|
|
|
|
bool release_backing = false;
|
|
|
|
bool release_address = true;
|
|
|
|
virErrorPtr orig_err = NULL;
|
|
|
|
virJSONValuePtr props = NULL;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-09-07 16:29:30 +00:00
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_SHMEM, { .shmem = shmem } };
|
2016-09-12 13:40:48 +00:00
|
|
|
|
|
|
|
switch ((virDomainShmemModel)shmem->model) {
|
|
|
|
case VIR_DOMAIN_SHMEM_MODEL_IVSHMEM_PLAIN:
|
|
|
|
case VIR_DOMAIN_SHMEM_MODEL_IVSHMEM_DOORBELL:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_SHMEM_MODEL_IVSHMEM:
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("live attach of shmem model '%s' is not supported"),
|
|
|
|
virDomainShmemModelTypeToString(shmem->model));
|
2017-06-07 08:46:41 +00:00
|
|
|
ATTRIBUTE_FALLTHROUGH;
|
2016-09-12 13:40:48 +00:00
|
|
|
case VIR_DOMAIN_SHMEM_MODEL_LAST:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (qemuAssignDeviceShmemAlias(vm->def, shmem, -1) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuDomainPrepareShmemChardev(shmem) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (VIR_REALLOC_N(vm->def->shmems, vm->def->nshmems + 1) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if ((shmem->info.type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE ||
|
|
|
|
shmem->info.type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI) &&
|
2016-11-03 20:33:32 +00:00
|
|
|
(qemuDomainEnsurePCIAddress(vm, &dev, driver) < 0))
|
2016-09-12 13:40:48 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (!(shmstr = qemuBuildShmemDevStr(vm->def, shmem, priv->qemuCaps)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (shmem->server.enabled) {
|
|
|
|
if (virAsprintf(&charAlias, "char%s", shmem->info.alias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
} else {
|
|
|
|
if (!(props = qemuBuildShmemBackendMemProps(shmem)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
|
|
|
if (shmem->server.enabled) {
|
|
|
|
if (qemuMonitorAttachCharDev(priv->mon, charAlias,
|
|
|
|
&shmem->server.chr) < 0)
|
|
|
|
goto exit_monitor;
|
|
|
|
} else {
|
2018-05-18 12:48:22 +00:00
|
|
|
if (qemuMonitorAddObject(priv->mon, &props, &memAlias) < 0)
|
2016-09-12 13:40:48 +00:00
|
|
|
goto exit_monitor;
|
|
|
|
}
|
|
|
|
|
|
|
|
release_backing = true;
|
|
|
|
|
2018-11-08 11:00:30 +00:00
|
|
|
if (qemuDomainAttachExtensionDevice(priv->mon, &shmem->info) < 0)
|
|
|
|
goto exit_monitor;
|
|
|
|
|
|
|
|
if (qemuMonitorAddDevice(priv->mon, shmstr) < 0) {
|
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, &shmem->info));
|
2016-09-12 13:40:48 +00:00
|
|
|
goto exit_monitor;
|
2018-11-08 11:00:30 +00:00
|
|
|
}
|
2016-09-12 13:40:48 +00:00
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
release_address = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Doing a copy here just so the pointer doesn't get nullified
|
|
|
|
* because we need it in the audit function */
|
|
|
|
VIR_APPEND_ELEMENT_COPY_INPLACE(vm->def->shmems, vm->def->nshmems, shmem);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
release_address = false;
|
|
|
|
|
|
|
|
audit:
|
|
|
|
virDomainAuditShmem(vm, shmem, "attach", ret == 0);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (release_address)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &shmem->info, NULL);
|
|
|
|
|
|
|
|
virJSONValueFree(props);
|
|
|
|
VIR_FREE(memAlias);
|
|
|
|
VIR_FREE(charAlias);
|
|
|
|
VIR_FREE(shmstr);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
exit_monitor:
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&orig_err);
|
2016-09-12 13:40:48 +00:00
|
|
|
if (release_backing) {
|
|
|
|
if (shmem->server.enabled)
|
|
|
|
ignore_value(qemuMonitorDetachCharDev(priv->mon, charAlias));
|
|
|
|
else
|
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, memAlias));
|
|
|
|
}
|
|
|
|
|
2017-02-22 17:39:17 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
release_address = false;
|
|
|
|
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&orig_err);
|
2016-09-12 13:40:48 +00:00
|
|
|
|
|
|
|
goto audit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-01 11:39:15 +00:00
|
|
|
int
|
|
|
|
qemuDomainAttachWatchdog(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainWatchdogDefPtr watchdog)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_WATCHDOG, { .watchdog = watchdog } };
|
|
|
|
virDomainWatchdogAction actualAction = watchdog->action;
|
|
|
|
const char *actionStr = NULL;
|
|
|
|
char *watchdogstr = NULL;
|
|
|
|
bool releaseAddress = false;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
if (vm->def->watchdog) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("domain already has a watchdog"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (qemuAssignDeviceWatchdogAlias(watchdog) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (watchdog->model == VIR_DOMAIN_WATCHDOG_MODEL_I6300ESB) {
|
|
|
|
if (qemuDomainEnsurePCIAddress(vm, &dev, driver) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
releaseAddress = true;
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("hotplug of watchdog of model %s is not supported"),
|
|
|
|
virDomainWatchdogModelTypeToString(watchdog->model));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
qemu: fix i6300esb watchdog hotplug on Q35
When commit 361c8dc17 added support for hotplugging the i6300esb
watchdog device (first in libvirt-3.9.0), it accidentally contstructed
the commandline for the device_add command before allocating a PCI
address for the device. With no PCI address specified in the command,
the watchdog would simply be placed at the lowest unused PCI slot.
On a 440fx guest, this doesn't cause a problem, because libvirt's PCI
address allocation algorithm would most likely give the same address
anyway (usually a slot on pci-root), so nobody noticed the omission of
address from the command.
But on a Q35 guest, the lowest unused PCI slot is on pcie-root, which
doesn't support hotplug; libvirt knows enough to assign a PCI address
that is on a pcie-to-pci-bridge (because its slots *do* support
hotplug), but qemu doesn't, so if there is no PCI address in the
command, qemu just tries to plug the new device into pcie-root, and
fails because it doesn't support hotplug, e.g.:
error: Failed to attach device from watchdog.xml
error: internal error: unable to execute QEMU command 'device_add':
Bus 'pcie.0' does not support hotplugging
The solution is simply to build the command string after assigning a
PCI address, not before.
Resolves: https://bugzilla.redhat.com/1666559
Signed-off-by: Laine Stump <laine@laine.org>
Reviewed-by: John Ferlan <jferlan@redhat.com>
2019-01-17 20:05:54 +00:00
|
|
|
if (!(watchdogstr = qemuBuildWatchdogDevStr(vm->def, watchdog, priv->qemuCaps)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2017-09-01 11:39:15 +00:00
|
|
|
/* QEMU doesn't have a 'dump' action; we tell qemu to 'pause', then
|
|
|
|
libvirt listens for the watchdog event, and we perform the dump
|
|
|
|
ourselves. so convert 'dump' to 'pause' for the qemu cli */
|
|
|
|
if (actualAction == VIR_DOMAIN_WATCHDOG_ACTION_DUMP)
|
|
|
|
actualAction = VIR_DOMAIN_WATCHDOG_ACTION_PAUSE;
|
|
|
|
|
|
|
|
actionStr = virDomainWatchdogActionTypeToString(actualAction);
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
|
|
|
rv = qemuMonitorSetWatchdogAction(priv->mon, actionStr);
|
|
|
|
|
|
|
|
if (rv >= 0)
|
|
|
|
rv = qemuMonitorAddDevice(priv->mon, watchdogstr);
|
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
releaseAddress = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rv < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
releaseAddress = false;
|
|
|
|
vm->def->watchdog = watchdog;
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (releaseAddress)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &watchdog->info, NULL);
|
|
|
|
VIR_FREE(watchdogstr);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-10-04 09:09:27 +00:00
|
|
|
int
|
|
|
|
qemuDomainAttachInputDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainInputDefPtr input)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
char *devstr = NULL;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_INPUT,
|
|
|
|
{ .input = input } };
|
2017-11-21 12:56:37 +00:00
|
|
|
virErrorPtr originalError = NULL;
|
2017-10-04 09:09:27 +00:00
|
|
|
bool releaseaddr = false;
|
2017-11-21 12:56:37 +00:00
|
|
|
bool teardowndevice = false;
|
|
|
|
bool teardownlabel = false;
|
|
|
|
bool teardowncgroup = false;
|
2017-10-04 09:09:27 +00:00
|
|
|
|
|
|
|
if (input->bus != VIR_DOMAIN_INPUT_BUS_USB &&
|
|
|
|
input->bus != VIR_DOMAIN_INPUT_BUS_VIRTIO) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("input device on bus '%s' cannot be hot plugged."),
|
|
|
|
virDomainInputBusTypeToString(input->bus));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (input->bus == VIR_DOMAIN_INPUT_BUS_VIRTIO) {
|
|
|
|
if (qemuDomainEnsureVirtioAddress(&releaseaddr, vm, &dev, "input") < 0)
|
|
|
|
return -1;
|
|
|
|
} else if (input->bus == VIR_DOMAIN_INPUT_BUS_USB) {
|
2017-10-20 11:28:21 +00:00
|
|
|
if (virDomainUSBAddressEnsure(priv->usbaddrs, &input->info) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
releaseaddr = true;
|
2017-10-04 09:09:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (qemuAssignDeviceInputAlias(vm->def, input, -1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuBuildInputDevStr(&devstr, vm->def, input, priv->qemuCaps) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2017-11-21 12:56:37 +00:00
|
|
|
if (qemuDomainNamespaceSetupInput(vm, input) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardowndevice = true;
|
|
|
|
|
|
|
|
if (qemuSetupInputCgroup(vm, input) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardowncgroup = true;
|
|
|
|
|
|
|
|
if (qemuSecuritySetInputLabel(vm, input) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
teardownlabel = true;
|
|
|
|
|
2017-10-04 09:09:27 +00:00
|
|
|
if (VIR_REALLOC_N(vm->def->inputs, vm->def->ninputs + 1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2018-11-08 11:00:30 +00:00
|
|
|
|
|
|
|
if (qemuDomainAttachExtensionDevice(priv->mon, &input->info) < 0)
|
2017-10-04 09:09:27 +00:00
|
|
|
goto exit_monitor;
|
|
|
|
|
2018-11-08 11:00:30 +00:00
|
|
|
if (qemuMonitorAddDevice(priv->mon, devstr) < 0) {
|
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, &input->info));
|
|
|
|
goto exit_monitor;
|
|
|
|
}
|
|
|
|
|
2017-10-04 09:09:27 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
releaseaddr = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_APPEND_ELEMENT_COPY_INPLACE(vm->def->inputs, vm->def->ninputs, input);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
audit:
|
|
|
|
virDomainAuditInput(vm, input, "attach", ret == 0);
|
|
|
|
|
|
|
|
cleanup:
|
2017-11-21 12:56:37 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
virErrorPreserveLast(&originalError);
|
|
|
|
if (teardownlabel)
|
|
|
|
qemuSecurityRestoreInputLabel(vm, input);
|
|
|
|
if (teardowncgroup)
|
|
|
|
qemuTeardownInputCgroup(vm, input);
|
|
|
|
if (teardowndevice)
|
|
|
|
qemuDomainNamespaceTeardownInput(vm, input);
|
|
|
|
if (releaseaddr)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &input->info, NULL);
|
|
|
|
virErrorRestore(&originalError);
|
|
|
|
}
|
2017-10-04 09:09:27 +00:00
|
|
|
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
exit_monitor:
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
releaseaddr = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
goto audit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-30 11:53:52 +00:00
|
|
|
int
|
|
|
|
qemuDomainAttachVsockDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainVsockDefPtr vsock)
|
|
|
|
{
|
|
|
|
qemuDomainVsockPrivatePtr vsockPriv = (qemuDomainVsockPrivatePtr)vsock->privateData;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virDomainDeviceDef dev = { VIR_DOMAIN_DEVICE_VSOCK,
|
|
|
|
{ .vsock = vsock } };
|
|
|
|
virErrorPtr originalError = NULL;
|
|
|
|
const char *fdprefix = "vsockfd";
|
|
|
|
bool releaseaddr = false;
|
|
|
|
char *fdname = NULL;
|
|
|
|
char *devstr = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (vm->def->vsock) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("the domain already has a vsock device"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (qemuDomainEnsureVirtioAddress(&releaseaddr, vm, &dev, "vsock") < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (qemuAssignDeviceVsockAlias(vsock) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuProcessOpenVhostVsock(vsock) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virAsprintf(&fdname, "%s%u", fdprefix, vsockPriv->vhostfd) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!(devstr = qemuBuildVsockDevStr(vm->def, vsock, priv->qemuCaps, fdprefix)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2018-11-08 11:00:30 +00:00
|
|
|
|
|
|
|
if (qemuDomainAttachExtensionDevice(priv->mon, &vsock->info) < 0)
|
|
|
|
goto exit_monitor;
|
|
|
|
|
|
|
|
if (qemuMonitorAddDeviceWithFd(priv->mon, devstr, vsockPriv->vhostfd, fdname) < 0) {
|
|
|
|
ignore_value(qemuDomainDetachExtensionDevice(priv->mon, &vsock->info));
|
2018-05-30 11:53:52 +00:00
|
|
|
goto exit_monitor;
|
2018-11-08 11:00:30 +00:00
|
|
|
}
|
2018-05-30 11:53:52 +00:00
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0) {
|
|
|
|
releaseaddr = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_STEAL_PTR(vm->def->vsock, vsock);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (ret < 0) {
|
|
|
|
virErrorPreserveLast(&originalError);
|
|
|
|
if (releaseaddr)
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &vsock->info, NULL);
|
|
|
|
virErrorRestore(&originalError);
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_FREE(devstr);
|
|
|
|
VIR_FREE(fdname);
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
exit_monitor:
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
releaseaddr = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
static int
|
2014-11-21 17:51:13 +00:00
|
|
|
qemuDomainChangeNetBridge(virDomainObjPtr vm,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
virDomainNetDefPtr olddev,
|
|
|
|
virDomainNetDefPtr newdev)
|
2012-03-28 19:11:09 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
2014-11-21 17:51:13 +00:00
|
|
|
const char *oldbridge = virDomainNetGetActualBridgeName(olddev);
|
|
|
|
const char *newbridge = virDomainNetGetActualBridgeName(newdev);
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
|
2014-11-21 17:51:13 +00:00
|
|
|
if (!oldbridge || !newbridge) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s", _("Missing bridge name"));
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
goto cleanup;
|
2014-11-21 17:51:13 +00:00
|
|
|
}
|
2012-03-28 19:11:09 +00:00
|
|
|
|
|
|
|
VIR_DEBUG("Change bridge for interface %s: %s -> %s",
|
|
|
|
olddev->ifname, oldbridge, newbridge);
|
|
|
|
|
|
|
|
if (virNetDevExists(newbridge) != 1) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("bridge %s doesn't exist"), newbridge);
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
goto cleanup;
|
2012-03-28 19:11:09 +00:00
|
|
|
}
|
|
|
|
|
2019-02-27 14:03:10 +00:00
|
|
|
ret = virNetDevBridgeRemovePort(oldbridge, olddev->ifname);
|
|
|
|
virDomainAuditNet(vm, olddev, NULL, "detach", ret == 0);
|
|
|
|
if (ret < 0) {
|
|
|
|
/* warn but continue - possibly the old network
|
|
|
|
* had been destroyed and reconstructed, leaving the
|
|
|
|
* tap device orphaned.
|
|
|
|
*/
|
|
|
|
VIR_WARN("Unable to detach device %s from bridge %s",
|
|
|
|
olddev->ifname, oldbridge);
|
2012-03-28 19:11:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = virNetDevBridgeAddPort(newbridge, olddev->ifname);
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
virDomainAuditNet(vm, NULL, newdev, "attach", ret == 0);
|
2012-03-28 19:11:09 +00:00
|
|
|
if (ret < 0) {
|
|
|
|
ret = virNetDevBridgeAddPort(oldbridge, olddev->ifname);
|
|
|
|
virDomainAuditNet(vm, NULL, olddev, "attach", ret == 0);
|
|
|
|
if (ret < 0) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
2012-09-12 16:54:42 +00:00
|
|
|
_("unable to recover former state by adding port "
|
2012-07-18 15:22:03 +00:00
|
|
|
"to bridge %s"), oldbridge);
|
2012-03-28 19:11:09 +00:00
|
|
|
}
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
goto cleanup;
|
2012-03-28 19:11:09 +00:00
|
|
|
}
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* caller will replace entire olddev with newdev in domain nets list */
|
|
|
|
ret = 0;
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
return ret;
|
2012-03-28 19:11:09 +00:00
|
|
|
}
|
|
|
|
|
2012-11-12 16:21:10 +00:00
|
|
|
static int
|
2014-11-07 10:37:37 +00:00
|
|
|
qemuDomainChangeNetFilter(virDomainObjPtr vm,
|
2012-11-12 16:21:10 +00:00
|
|
|
virDomainNetDefPtr olddev,
|
|
|
|
virDomainNetDefPtr newdev)
|
|
|
|
{
|
|
|
|
/* make sure this type of device supports filters. */
|
|
|
|
switch (virDomainNetGetActualType(newdev)) {
|
|
|
|
case VIR_DOMAIN_NET_TYPE_ETHERNET:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_BRIDGE:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_NETWORK:
|
|
|
|
break;
|
2018-02-14 09:43:59 +00:00
|
|
|
case VIR_DOMAIN_NET_TYPE_USER:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_VHOSTUSER:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_SERVER:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_CLIENT:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_MCAST:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_INTERNAL:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_DIRECT:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_HOSTDEV:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_UDP:
|
2012-11-12 16:21:10 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("filters not supported on interfaces of type %s"),
|
|
|
|
virDomainNetTypeToString(virDomainNetGetActualType(newdev)));
|
|
|
|
return -1;
|
2018-02-14 09:43:59 +00:00
|
|
|
case VIR_DOMAIN_NET_TYPE_LAST:
|
|
|
|
default:
|
|
|
|
virReportEnumRangeError(virDomainNetType,
|
|
|
|
virDomainNetGetActualType(newdev));
|
|
|
|
return -1;
|
2012-11-12 16:21:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virDomainConfNWFilterTeardown(olddev);
|
|
|
|
|
2014-05-01 08:40:41 +00:00
|
|
|
if (newdev->filter &&
|
2018-04-27 13:28:11 +00:00
|
|
|
virDomainConfNWFilterInstantiate(vm->def->name,
|
2018-05-11 17:39:27 +00:00
|
|
|
vm->def->uuid, newdev, false) < 0) {
|
2012-11-12 16:21:10 +00:00
|
|
|
virErrorPtr errobj;
|
|
|
|
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("failed to add new filter rules to '%s' "
|
|
|
|
"- attempting to restore old rules"),
|
|
|
|
olddev->ifname);
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorPreserveLast(&errobj);
|
2018-04-27 13:28:11 +00:00
|
|
|
ignore_value(virDomainConfNWFilterInstantiate(vm->def->name,
|
2018-05-11 17:39:27 +00:00
|
|
|
vm->def->uuid, olddev, false));
|
2017-09-12 08:32:27 +00:00
|
|
|
virErrorRestore(&errobj);
|
2012-11-12 16:21:10 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-28 16:43:10 +00:00
|
|
|
int qemuDomainChangeNetLinkState(virQEMUDriverPtr driver,
|
2011-09-06 08:23:47 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainNetDefPtr dev,
|
|
|
|
int linkstate)
|
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
|
|
|
|
if (!dev->info.alias) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("can't change link state: device alias not found"));
|
2011-09-06 08:23:47 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-09-24 12:30:09 +00:00
|
|
|
VIR_DEBUG("dev: %s, state: %d", dev->info.alias, linkstate);
|
|
|
|
|
2013-02-06 18:17:20 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2011-09-06 08:23:47 +00:00
|
|
|
|
|
|
|
ret = qemuMonitorSetLink(priv->mon, dev->info.alias, linkstate);
|
|
|
|
if (ret < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* modify the device configuration */
|
|
|
|
dev->linkstate = linkstate;
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2015-01-07 12:12:18 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
return -1;
|
2011-09-06 08:23:47 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
int
|
2012-11-28 16:43:10 +00:00
|
|
|
qemuDomainChangeNet(virQEMUDriverPtr driver,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDeviceDefPtr dev)
|
2011-09-06 08:23:47 +00:00
|
|
|
{
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
virDomainNetDefPtr newdev = dev->data.net;
|
2015-06-08 08:25:10 +00:00
|
|
|
virDomainNetDefPtr *devslot = NULL;
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
virDomainNetDefPtr olddev;
|
2016-09-23 15:04:53 +00:00
|
|
|
virDomainNetType oldType, newType;
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
bool needReconnect = false;
|
|
|
|
bool needBridgeChange = false;
|
2012-11-12 16:21:10 +00:00
|
|
|
bool needFilterChange = false;
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
bool needLinkStateChange = false;
|
|
|
|
bool needReplaceDevDef = false;
|
2013-10-01 13:04:48 +00:00
|
|
|
bool needBandwidthSet = false;
|
2017-06-15 12:22:26 +00:00
|
|
|
bool needCoalesceChange = false;
|
2017-07-17 15:49:00 +00:00
|
|
|
bool needVlanUpdate = false;
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
int ret = -1;
|
2015-06-08 08:25:10 +00:00
|
|
|
int changeidx = -1;
|
|
|
|
|
|
|
|
if ((changeidx = virDomainNetFindIdx(vm->def, newdev)) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
devslot = &vm->def->nets[changeidx];
|
2018-02-22 12:24:58 +00:00
|
|
|
olddev = *devslot;
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
|
|
|
|
oldType = virDomainNetGetActualType(olddev);
|
|
|
|
if (oldType == VIR_DOMAIN_NET_TYPE_HOSTDEV) {
|
|
|
|
/* no changes are possible to a type='hostdev' interface */
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot change config of '%s' network type"),
|
|
|
|
virDomainNetTypeToString(oldType));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check individual attributes for changes that can't be done to a
|
|
|
|
* live netdev. These checks *mostly* go in order of the
|
|
|
|
* declarations in virDomainNetDef in order to assure nothing is
|
|
|
|
* omitted. (exceptiong where noted in comments - in particular,
|
|
|
|
* some things require that a new "actual device" be allocated
|
|
|
|
* from the network driver first, but we delay doing that until
|
|
|
|
* after we've made as many other checks as possible)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* type: this can change (with some restrictions), but the actual
|
|
|
|
* type of the new device connection isn't known until after we
|
|
|
|
* allocate the "actual" device.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (virMacAddrCmp(&olddev->mac, &newdev->mac)) {
|
|
|
|
char oldmac[VIR_MAC_STRING_BUFLEN], newmac[VIR_MAC_STRING_BUFLEN];
|
|
|
|
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot change network interface mac address "
|
|
|
|
"from %s to %s"),
|
|
|
|
virMacAddrFormat(&olddev->mac, oldmac),
|
|
|
|
virMacAddrFormat(&newdev->mac, newmac));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STRNEQ_NULLABLE(olddev->model, newdev->model)) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot modify network device model from %s to %s"),
|
|
|
|
olddev->model ? olddev->model : "(default)",
|
|
|
|
newdev->model ? newdev->model : "(default)");
|
|
|
|
goto cleanup;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
|
|
|
|
2019-01-21 22:59:02 +00:00
|
|
|
if (virDomainNetIsVirtioModel(olddev) &&
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
(olddev->driver.virtio.name != newdev->driver.virtio.name ||
|
|
|
|
olddev->driver.virtio.txmode != newdev->driver.virtio.txmode ||
|
|
|
|
olddev->driver.virtio.ioeventfd != newdev->driver.virtio.ioeventfd ||
|
2013-12-03 10:40:38 +00:00
|
|
|
olddev->driver.virtio.event_idx != newdev->driver.virtio.event_idx ||
|
2014-10-30 12:32:00 +00:00
|
|
|
olddev->driver.virtio.queues != newdev->driver.virtio.queues ||
|
2017-08-31 09:33:06 +00:00
|
|
|
olddev->driver.virtio.rx_queue_size != newdev->driver.virtio.rx_queue_size ||
|
|
|
|
olddev->driver.virtio.tx_queue_size != newdev->driver.virtio.tx_queue_size ||
|
2014-10-30 12:32:00 +00:00
|
|
|
olddev->driver.virtio.host.csum != newdev->driver.virtio.host.csum ||
|
|
|
|
olddev->driver.virtio.host.gso != newdev->driver.virtio.host.gso ||
|
|
|
|
olddev->driver.virtio.host.tso4 != newdev->driver.virtio.host.tso4 ||
|
|
|
|
olddev->driver.virtio.host.tso6 != newdev->driver.virtio.host.tso6 ||
|
|
|
|
olddev->driver.virtio.host.ecn != newdev->driver.virtio.host.ecn ||
|
|
|
|
olddev->driver.virtio.host.ufo != newdev->driver.virtio.host.ufo ||
|
2015-02-06 14:40:19 +00:00
|
|
|
olddev->driver.virtio.host.mrg_rxbuf != newdev->driver.virtio.host.mrg_rxbuf ||
|
2014-10-30 12:32:00 +00:00
|
|
|
olddev->driver.virtio.guest.csum != newdev->driver.virtio.guest.csum ||
|
|
|
|
olddev->driver.virtio.guest.tso4 != newdev->driver.virtio.guest.tso4 ||
|
|
|
|
olddev->driver.virtio.guest.tso6 != newdev->driver.virtio.guest.tso6 ||
|
|
|
|
olddev->driver.virtio.guest.ecn != newdev->driver.virtio.guest.ecn ||
|
|
|
|
olddev->driver.virtio.guest.ufo != newdev->driver.virtio.guest.ufo)) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot modify virtio network device driver attributes"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* data: this union will be examined later, after allocating new actualdev */
|
|
|
|
/* virtPortProfile: will be examined later, after allocating new actualdev */
|
|
|
|
|
|
|
|
if (olddev->tune.sndbuf_specified != newdev->tune.sndbuf_specified ||
|
|
|
|
olddev->tune.sndbuf != newdev->tune.sndbuf) {
|
|
|
|
needReconnect = true;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (STRNEQ_NULLABLE(olddev->script, newdev->script)) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot modify network device script attribute"));
|
|
|
|
goto cleanup;
|
2012-07-30 06:03:25 +00:00
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* ifname: check if it's set in newdev. If not, retain the autogenerated one */
|
2013-05-20 09:23:13 +00:00
|
|
|
if (!newdev->ifname && VIR_STRDUP(newdev->ifname, olddev->ifname) < 0)
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
goto cleanup;
|
|
|
|
if (STRNEQ_NULLABLE(olddev->ifname, newdev->ifname)) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot modify network device tap name"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2011-09-06 08:23:47 +00:00
|
|
|
|
2018-08-24 10:28:41 +00:00
|
|
|
/* info: Nothing is allowed to change. First fill the missing newdev->info
|
|
|
|
* from olddev and then check for changes.
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
*/
|
2018-08-24 10:28:41 +00:00
|
|
|
/* if pci addr is missing or is invalid we overwrite it from olddev */
|
|
|
|
if (newdev->info.type == VIR_DOMAIN_DEVICE_ADDRESS_TYPE_NONE ||
|
|
|
|
!virDomainDeviceAddressIsValid(&newdev->info,
|
|
|
|
newdev->info.type)) {
|
|
|
|
newdev->info.type = olddev->info.type;
|
|
|
|
newdev->info.addr = olddev->info.addr;
|
|
|
|
}
|
|
|
|
if (olddev->info.type != newdev->info.type) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("cannot modify network device address type"));
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
}
|
2016-04-03 18:16:51 +00:00
|
|
|
if (!virPCIDeviceAddressEqual(&olddev->info.addr.pci,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
&newdev->info.addr.pci)) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot modify network device guest PCI address"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
/* grab alias from olddev if not set in newdev */
|
2013-05-20 09:23:13 +00:00
|
|
|
if (!newdev->info.alias &&
|
|
|
|
VIR_STRDUP(newdev->info.alias, olddev->info.alias) < 0)
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
goto cleanup;
|
2018-06-12 14:05:10 +00:00
|
|
|
|
|
|
|
/* device alias is checked already in virDomainDefCompatibleDevice */
|
|
|
|
|
2018-08-24 10:28:41 +00:00
|
|
|
if (newdev->info.rombar == VIR_TRISTATE_BOOL_ABSENT)
|
|
|
|
newdev->info.rombar = olddev->info.rombar;
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (olddev->info.rombar != newdev->info.rombar) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot modify network device rom bar setting"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-08-24 10:28:41 +00:00
|
|
|
|
|
|
|
if (!newdev->info.romfile &&
|
|
|
|
VIR_STRDUP(newdev->info.romfile, olddev->info.romfile) < 0)
|
|
|
|
goto cleanup;
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (STRNEQ_NULLABLE(olddev->info.romfile, newdev->info.romfile)) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot modify network rom file"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-08-24 10:28:41 +00:00
|
|
|
|
|
|
|
if (newdev->info.bootIndex == 0)
|
|
|
|
newdev->info.bootIndex = olddev->info.bootIndex;
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (olddev->info.bootIndex != newdev->info.bootIndex) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot modify network device boot index setting"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2018-08-24 10:28:41 +00:00
|
|
|
|
|
|
|
if (newdev->info.romenabled == VIR_TRISTATE_BOOL_ABSENT)
|
|
|
|
newdev->info.romenabled = olddev->info.romenabled;
|
2018-07-13 13:07:58 +00:00
|
|
|
if (olddev->info.romenabled != newdev->info.romenabled) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("cannot modify network device rom enabled setting"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* (end of device info checks) */
|
2011-09-06 08:23:47 +00:00
|
|
|
|
2012-11-12 16:21:10 +00:00
|
|
|
if (STRNEQ_NULLABLE(olddev->filter, newdev->filter) ||
|
|
|
|
!virNWFilterHashTableEqual(olddev->filterparams, newdev->filterparams)) {
|
|
|
|
needFilterChange = true;
|
|
|
|
}
|
2011-09-06 08:23:47 +00:00
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* bandwidth can be modified, and will be checked later */
|
|
|
|
/* vlan can be modified, and will be checked later */
|
|
|
|
/* linkstate can be modified */
|
|
|
|
|
2017-06-08 11:45:31 +00:00
|
|
|
if (olddev->mtu != newdev->mtu) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("cannot modify MTU"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* allocate new actual device to compare to old - we will need to
|
|
|
|
* free it if we fail for any reason
|
|
|
|
*/
|
|
|
|
if (newdev->type == VIR_DOMAIN_NET_TYPE_NETWORK &&
|
2018-01-25 09:35:47 +00:00
|
|
|
virDomainNetAllocateActualDevice(vm->def, newdev) < 0) {
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
newType = virDomainNetGetActualType(newdev);
|
|
|
|
|
|
|
|
if (newType == VIR_DOMAIN_NET_TYPE_HOSTDEV) {
|
|
|
|
/* can't turn it into a type='hostdev' interface */
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("cannot change network interface type to '%s'"),
|
|
|
|
virDomainNetTypeToString(newType));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (olddev->type == newdev->type && oldType == newType) {
|
2011-09-06 08:23:47 +00:00
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* if type hasn't changed, check the relevant fields for the type */
|
|
|
|
switch (newdev->type) {
|
|
|
|
case VIR_DOMAIN_NET_TYPE_USER:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_ETHERNET:
|
conf/openvz: eliminate incorrect/undocumented use of <source dev='blah'/>
When support for <interface type='ethernet'> was added in commit
9a4b705f back in 2010, it erroneously looked at <source dev='blah'/>
for a user-specified guest-side interface name. This was never
documented though. (that attribute already existed at the time in the
data.ethernet union member of virDomainNetDef, but apparently had no
practical use - it was only used as a storage place for a NetDef's
bridge name during qemuDomainXMLToNative(), but even then that was
never used for anything).
When support for similar guest-side device naming was added to the lxc
driver several years later, it was put in a new subelement <guest
dev='blah'/>.
In the intervening years, since there was no validation that
ethernet.dev was NULL in the other drivers that didn't actually use
it, innocent souls who were adding other features assuming they needed
to account for non-NULL ethernet.dev when really they didn't, so
little bits of the usual pointless cargo-cult code showed up.
This patch not only switches the openvz driver to use the documented
<guest dev='blah'/> notation for naming the guest-side device (just in
case anyone is still using the openvz driver), and logs an error if
anyone tries to set <source dev='blah'/> for a type='ethernet'
interface, it also removes the cargo-cult uses of ethernet.dev and
<source dev='blah'/>, and eliminates if from the RNG and from
virDomainNetDef.
NB: I decided on this course of action after mentioning the
inconsistency here:
https://www.redhat.com/archives/libvir-list/2016-May/msg02038.html
and getting encouragement do eliminate it in a later IRC discussion
with danpb.
2016-06-21 19:20:57 +00:00
|
|
|
break;
|
2011-09-06 08:23:47 +00:00
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
case VIR_DOMAIN_NET_TYPE_SERVER:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_CLIENT:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_MCAST:
|
2015-08-29 20:19:10 +00:00
|
|
|
case VIR_DOMAIN_NET_TYPE_UDP:
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (STRNEQ_NULLABLE(olddev->data.socket.address,
|
|
|
|
newdev->data.socket.address) ||
|
|
|
|
olddev->data.socket.port != newdev->data.socket.port) {
|
|
|
|
needReconnect = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_NETWORK:
|
|
|
|
if (STRNEQ(olddev->data.network.name, newdev->data.network.name)) {
|
|
|
|
if (virDomainNetGetActualVirtPortProfile(newdev))
|
|
|
|
needReconnect = true;
|
|
|
|
else
|
|
|
|
needBridgeChange = true;
|
|
|
|
}
|
|
|
|
/* other things handled in common code directly below this switch */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_BRIDGE:
|
|
|
|
/* all handled in bridge name checked in common code below */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_INTERNAL:
|
|
|
|
if (STRNEQ_NULLABLE(olddev->data.internal.name,
|
|
|
|
newdev->data.internal.name)) {
|
|
|
|
needReconnect = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_NET_TYPE_DIRECT:
|
|
|
|
/* all handled in common code directly below this switch */
|
|
|
|
break;
|
|
|
|
|
2018-02-14 09:43:59 +00:00
|
|
|
case VIR_DOMAIN_NET_TYPE_VHOSTUSER:
|
|
|
|
case VIR_DOMAIN_NET_TYPE_HOSTDEV:
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("unable to change config on '%s' network type"),
|
|
|
|
virDomainNetTypeToString(newdev->type));
|
2018-02-14 09:43:59 +00:00
|
|
|
goto cleanup;
|
|
|
|
case VIR_DOMAIN_NET_TYPE_LAST:
|
|
|
|
default:
|
|
|
|
virReportEnumRangeError(virDomainNetType, newdev->type);
|
|
|
|
goto cleanup;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
} else {
|
|
|
|
/* interface type has changed. There are a few special cases
|
|
|
|
* where this can only require a minor (or even no) change,
|
|
|
|
* but in most cases we need to do a full reconnection.
|
|
|
|
*
|
|
|
|
* If we switch (in either direction) between type='bridge'
|
|
|
|
* and type='network' (for a traditional managed virtual
|
|
|
|
* network that uses a host bridge, i.e. forward
|
|
|
|
* mode='route|nat'), we just need to change the bridge.
|
|
|
|
*/
|
|
|
|
if ((oldType == VIR_DOMAIN_NET_TYPE_NETWORK &&
|
|
|
|
newType == VIR_DOMAIN_NET_TYPE_BRIDGE) ||
|
|
|
|
(oldType == VIR_DOMAIN_NET_TYPE_BRIDGE &&
|
|
|
|
newType == VIR_DOMAIN_NET_TYPE_NETWORK)) {
|
|
|
|
|
|
|
|
needBridgeChange = true;
|
|
|
|
|
|
|
|
} else if (oldType == VIR_DOMAIN_NET_TYPE_DIRECT &&
|
|
|
|
newType == VIR_DOMAIN_NET_TYPE_DIRECT) {
|
|
|
|
|
|
|
|
/* this is the case of switching from type='direct' to
|
|
|
|
* type='network' for a network that itself uses direct
|
|
|
|
* (macvtap) devices. If the physical device and mode are
|
|
|
|
* the same, this doesn't require any actual setup
|
|
|
|
* change. If the physical device or mode *does* change,
|
|
|
|
* that will be caught in the common section below */
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
/* for all other combinations, we'll need a full reconnect */
|
|
|
|
needReconnect = true;
|
2011-09-06 08:23:47 +00:00
|
|
|
|
|
|
|
}
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
}
|
2011-09-06 08:23:47 +00:00
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* now several things that are in multiple (but not all)
|
|
|
|
* different types, and can be safely compared even for those
|
|
|
|
* cases where they don't apply to a particular type.
|
|
|
|
*/
|
|
|
|
if (STRNEQ_NULLABLE(virDomainNetGetActualBridgeName(olddev),
|
|
|
|
virDomainNetGetActualBridgeName(newdev))) {
|
|
|
|
if (virDomainNetGetActualVirtPortProfile(newdev))
|
|
|
|
needReconnect = true;
|
|
|
|
else
|
|
|
|
needBridgeChange = true;
|
|
|
|
}
|
2011-09-06 08:23:47 +00:00
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (STRNEQ_NULLABLE(virDomainNetGetActualDirectDev(olddev),
|
|
|
|
virDomainNetGetActualDirectDev(newdev)) ||
|
2017-04-25 06:16:20 +00:00
|
|
|
virDomainNetGetActualDirectMode(olddev) != virDomainNetGetActualDirectMode(newdev) ||
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
!virNetDevVPortProfileEqual(virDomainNetGetActualVirtPortProfile(olddev),
|
2017-07-17 15:49:00 +00:00
|
|
|
virDomainNetGetActualVirtPortProfile(newdev))) {
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
needReconnect = true;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
|
|
|
|
2017-07-17 15:49:00 +00:00
|
|
|
if (!virNetDevVlanEqual(virDomainNetGetActualVlan(olddev),
|
|
|
|
virDomainNetGetActualVlan(newdev))) {
|
|
|
|
needVlanUpdate = true;
|
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (olddev->linkstate != newdev->linkstate)
|
|
|
|
needLinkStateChange = true;
|
|
|
|
|
2013-10-01 13:04:48 +00:00
|
|
|
if (!virNetDevBandwidthEqual(virDomainNetGetActualBandwidth(olddev),
|
|
|
|
virDomainNetGetActualBandwidth(newdev)))
|
|
|
|
needBandwidthSet = true;
|
|
|
|
|
2017-06-15 12:22:26 +00:00
|
|
|
if (!!olddev->coalesce != !!newdev->coalesce ||
|
|
|
|
(olddev->coalesce && newdev->coalesce &&
|
2017-06-21 07:00:58 +00:00
|
|
|
memcmp(olddev->coalesce, newdev->coalesce,
|
|
|
|
sizeof(*olddev->coalesce))))
|
2017-06-15 12:22:26 +00:00
|
|
|
needCoalesceChange = true;
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* FINALLY - actually perform the required actions */
|
|
|
|
|
|
|
|
if (needReconnect) {
|
2013-03-20 15:57:08 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
_("unable to change config on '%s' network type"),
|
|
|
|
virDomainNetTypeToString(newdev->type));
|
|
|
|
goto cleanup;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
|
|
|
|
2013-10-01 13:04:48 +00:00
|
|
|
if (needBandwidthSet) {
|
qemu: delete exist bandwidth restrictions when they are removed from config
When the <bandwidth> of an interface is changed with update-device,
the old settings are cleared with tc, then new settings added with
tc. But if the <bandwidth has been removed, the old settings weren't
being removed, so the bandwidth restrictions would still be active on
the interface although the interface status in libvirt showed that
they had been removed.
This patch fixes it by calling virNetDevBandwidthClear() if the
"modification" to the interface bandwidth was to completely clear
it.
An alternative could have been to modify virNetDevBandwidthSet() to
always clear existing bandwith settings at the beginning of the
function (currently it short circuits in that case, doing nothing),
but that would have led to cases where virNetDevBandwidthClear() was
now being called in cases where it previously wasn't, and while many
of those cases would be NOPs, there could be cases where it would
cause an error. The way this patch works, the ...Clear() function is
only called in cases where the ...Set() function had previously been
called successfully, so the risk of regression is minimized.
Resolves: https://bugzilla.redhat.com/1454709
2017-12-11 19:26:54 +00:00
|
|
|
virNetDevBandwidthPtr newb = virDomainNetGetActualBandwidth(newdev);
|
|
|
|
|
|
|
|
if (newb) {
|
|
|
|
if (virNetDevBandwidthSet(newdev->ifname, newb, false,
|
|
|
|
!virDomainNetTypeSharesHostView(newdev)) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* virNetDevBandwidthSet() doesn't clear any existing
|
|
|
|
* setting unless something new is being set.
|
|
|
|
*/
|
|
|
|
virNetDevBandwidthClear(newdev->ifname);
|
|
|
|
}
|
2013-10-01 13:04:48 +00:00
|
|
|
needReplaceDevDef = true;
|
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (needBridgeChange) {
|
2014-11-21 17:51:13 +00:00
|
|
|
if (qemuDomainChangeNetBridge(vm, olddev, newdev) < 0)
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
goto cleanup;
|
|
|
|
/* we successfully switched to the new bridge, and we've
|
|
|
|
* determined that the rest of newdev is equivalent to olddev,
|
2012-11-12 16:21:10 +00:00
|
|
|
* so move newdev into place */
|
|
|
|
needReplaceDevDef = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (needFilterChange) {
|
2014-11-07 10:37:37 +00:00
|
|
|
if (qemuDomainChangeNetFilter(vm, olddev, newdev) < 0)
|
2012-11-12 16:21:10 +00:00
|
|
|
goto cleanup;
|
|
|
|
/* we successfully switched to the new filter, and we've
|
|
|
|
* determined that the rest of newdev is equivalent to olddev,
|
|
|
|
* so move newdev into place */
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
needReplaceDevDef = true;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
|
|
|
|
2017-06-15 12:22:26 +00:00
|
|
|
if (needCoalesceChange) {
|
|
|
|
if (virNetDevSetCoalesce(newdev->ifname, newdev->coalesce, true) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
needReplaceDevDef = true;
|
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (needLinkStateChange &&
|
|
|
|
qemuDomainChangeNetLinkState(driver, vm, olddev, newdev->linkstate) < 0) {
|
|
|
|
goto cleanup;
|
2012-03-28 19:11:09 +00:00
|
|
|
}
|
|
|
|
|
2017-07-17 15:49:00 +00:00
|
|
|
if (needVlanUpdate) {
|
|
|
|
if (virNetDevOpenvswitchUpdateVlan(newdev->ifname, &newdev->vlan) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
needReplaceDevDef = true;
|
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
if (needReplaceDevDef) {
|
|
|
|
/* the changes above warrant replacing olddev with newdev in
|
|
|
|
* the domain's nets list.
|
|
|
|
*/
|
2013-08-27 17:06:18 +00:00
|
|
|
|
|
|
|
/* this function doesn't work with HOSTDEV networks yet, thus
|
|
|
|
* no need to change the pointer in the hostdev structure */
|
2018-01-25 09:35:47 +00:00
|
|
|
virDomainNetReleaseActualDevice(vm->def, olddev);
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
virDomainNetDefFree(olddev);
|
|
|
|
/* move newdev into the nets list, and NULL it out from the
|
|
|
|
* virDomainDeviceDef that we were given so that the caller
|
|
|
|
* won't delete it on return.
|
|
|
|
*/
|
|
|
|
*devslot = newdev;
|
|
|
|
newdev = dev->data.net = NULL;
|
|
|
|
dev->type = VIR_DOMAIN_DEVICE_NONE;
|
2011-09-06 08:23:47 +00:00
|
|
|
}
|
|
|
|
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
ret = 0;
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
/* When we get here, we will be in one of these two states:
|
|
|
|
*
|
|
|
|
* 1) newdev has been moved into the domain's list of nets and
|
|
|
|
* newdev set to NULL, and dev->data.net will be NULL (and
|
|
|
|
* dev->type is NONE). olddev will have been completely
|
|
|
|
* released and freed. (aka success) In this case no extra
|
|
|
|
* cleanup is needed.
|
|
|
|
*
|
|
|
|
* 2) newdev has *not* been moved into the domain's list of nets,
|
|
|
|
* and dev->data.net == newdev (and dev->type == NET). In this *
|
|
|
|
* case, we need to at least release the "actual device" from *
|
|
|
|
* newdev (the caller will free dev->data.net a.k.a. newdev, and
|
|
|
|
* the original olddev is still in used)
|
|
|
|
*
|
|
|
|
* Note that case (2) isn't necessarily a failure. It may just be
|
|
|
|
* that the changes were minor enough that we didn't need to
|
|
|
|
* replace the entire device object.
|
|
|
|
*/
|
|
|
|
if (newdev)
|
2018-01-25 09:35:47 +00:00
|
|
|
virDomainNetReleaseActualDevice(vm->def, newdev);
|
qemu: reorganize qemuDomainChangeNet and qemuDomainChangeNetBridge
This patch resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=805071
to the extent that it can be resolved with current qemu functionality.
It attempts to detect as many situations as possible when the simple
operation of disconnecting an existing tap device from one bridge and
attaching it to another will satisfy the change requested in
virDomainUpdateDeviceFlags() for a network device. Before this patch,
that situation could only be detected if the pre-change interface
*and* the post-change interface definition were both "type='bridge'".
After this patch, it can also be detected if the before or after
interfaces are any combination of type='bridge' and type='network'
(the networks can be <forward mode='nat|route|bridge'>, as long as
they use a Linux host bridge and not macvtap connections).
This extra effort is especially useful since the recent discovery that
a netdev_del+netdev_add combo (to reconnect the network device with
completely different hostside configuration) doesn't work properly
with current qemu (1.2) unless it is accompanied by the matching
device_del+device_add - see this mailing list message for details:
http://lists.nongnu.org/archive/html/qemu-devel/2012-10/msg02355.html
(A slight modification of the patch referenced there has been prepared
to apply on top of this patch, but won't be pushed until qemu can be
made to work with it.)
* qemuDomainChangeNet needs access to the virDomainDeviceDef that
holds the new netdef (so that it can clear out the virDomainDeviceDef
if it ends up using the NetDef to replace the original), so the
virDomainNetDefPtr arg is replaced with a virDomainDeviceDefPtr.
* qemuDomainChangeNet previously checked for *some* changes to the
interface config, but this check was by no means complete. It was also
a bit disorganized.
This refactoring of the code is (I believe) complete in its check of
all NetDef attributes that might be changed, and either returns a
failure (for changes that are simply impossible), or sets one of three
flags:
needLinkStateChange - if the device link state needs to go up/down
needBridgeChange - if everything else is the same, but it needs
to be connected to a difference linux host
bridge
needReconnect - if the entire host side of the device needs
to be torn down and reconstructed (currently
non-working, as mentioned above)
Note that this function will refuse to make any change that requires
the *guest* side of the device to be detached (e.g. changing the PCI
address or mac address). Those would be disruptive enough to the guest
that it's reasonable to require an explicit detach/attach sequence
from the management application.
* As mentioned above, qemuDomainChangeNet also does its best to
understand when a simple change in attached bridge for the existing
tap device will work vs. the need to completely tear down/reconstruct
the host side of the device (including tap device).
This patch *does not* implement the "reconnect" code anyway - there is
a placeholder that turns that into an error. Rather, the purpose of
this patch is to replicate existing behavior with code that is ready
to have that functionality plugged in in a later patch.
* The expanded uses for qemuDomainChangeNetBridge meant that it needed
to be enhanced as well - it no longer replaces the original brname
string in olddev with the new brname; instead, it relies on the
caller to replace the *entire* olddev with newdev (since we've gone
to great lengths to assure they are functionally identical other
than the name of the bridge, this is now not only safe, but more
correct). Additionally, qemuDomainNetChangeBridge can now set the
bridge for type='network' interfaces as well as plain type='bridge'
interfaces. (Note that I had to make this change simultaneous to the
reorganization of qemuDomainChangeNet because the two are too
closely intertwined to separate).
2012-10-10 19:38:00 +00:00
|
|
|
|
2011-09-06 08:23:47 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2014-12-08 13:48:32 +00:00
|
|
|
static virDomainGraphicsDefPtr
|
|
|
|
qemuDomainFindGraphics(virDomainObjPtr vm,
|
|
|
|
virDomainGraphicsDefPtr dev)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
size_t i;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2013-05-21 07:21:20 +00:00
|
|
|
for (i = 0; i < vm->def->ngraphics; i++) {
|
2010-12-16 16:10:54 +00:00
|
|
|
if (vm->def->graphics[i]->type == dev->type)
|
|
|
|
return vm->def->graphics[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-12-08 13:48:33 +00:00
|
|
|
int
|
|
|
|
qemuDomainFindGraphicsIndex(virDomainDefPtr def,
|
|
|
|
virDomainGraphicsDefPtr dev)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ngraphics; i++) {
|
|
|
|
if (def->graphics[i]->type == dev->type)
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-12-16 16:10:54 +00:00
|
|
|
int
|
2012-11-28 16:43:10 +00:00
|
|
|
qemuDomainChangeGraphics(virQEMUDriverPtr driver,
|
2010-12-16 16:10:54 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainGraphicsDefPtr dev)
|
|
|
|
{
|
|
|
|
virDomainGraphicsDefPtr olddev = qemuDomainFindGraphics(vm, dev);
|
2013-01-10 21:03:14 +00:00
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2016-04-25 13:36:28 +00:00
|
|
|
const char *type = virDomainGraphicsTypeToString(dev->type);
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
size_t i;
|
2016-04-25 13:36:28 +00:00
|
|
|
int ret = -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
if (!olddev) {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2018-01-23 04:24:42 +00:00
|
|
|
_("cannot find existing graphics device to modify of "
|
|
|
|
"type '%s'"), type);
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
if (dev->nListens != olddev->nListens) {
|
2016-04-25 13:36:28 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("cannot change the number of listen addresses "
|
|
|
|
"on '%s' graphics"), type);
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < dev->nListens; i++) {
|
2013-06-20 18:55:00 +00:00
|
|
|
virDomainGraphicsListenDefPtr newlisten = &dev->listens[i];
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
virDomainGraphicsListenDefPtr oldlisten = &olddev->listens[i];
|
|
|
|
|
2013-06-20 18:55:00 +00:00
|
|
|
if (newlisten->type != oldlisten->type) {
|
2016-04-25 13:36:28 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("cannot change the type of listen address "
|
|
|
|
"on '%s' graphics"), type);
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2016-05-02 15:45:23 +00:00
|
|
|
switch (newlisten->type) {
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS:
|
2013-06-20 18:55:00 +00:00
|
|
|
if (STRNEQ_NULLABLE(newlisten->address, oldlisten->address)) {
|
2016-04-25 13:36:28 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("cannot change listen address setting "
|
|
|
|
"on '%s' graphics"), type);
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2016-04-25 13:36:28 +00:00
|
|
|
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK:
|
2013-06-20 18:55:00 +00:00
|
|
|
if (STRNEQ_NULLABLE(newlisten->network, oldlisten->network)) {
|
2016-04-25 13:36:28 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("cannot change listen address setting "
|
|
|
|
"on '%s' graphics"), type);
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2016-04-25 13:36:28 +00:00
|
|
|
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
break;
|
|
|
|
|
2016-06-08 08:35:37 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_SOCKET:
|
|
|
|
if (STRNEQ_NULLABLE(newlisten->socket, oldlisten->socket)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("cannot change listen socket setting "
|
|
|
|
"on '%s' graphics"), type);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
qemuDomainChangeGraphics: Check listen address change by listen type
Currently, we have a bug when updating a graphics device. A graphics device can
have a listen address set. This address is either defined by user (in which case
it's type is VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_ADDRESS) or it can be inherited
from a network (in which case it's type is
VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NETWORK). However, in both cases we have a
listen address to process (e.g. during migration, as I've tried to fix in
7f15ebc7).
Later, when a user tries to update the graphics device (e.g. set a password),
we check if listen addresses match the original as qemu doesn't know how to
change listen address yet. Hence, users are required to not change the listen
address. The implementation then just dumps listen addresses and compare them.
Previously, while dumping the listen addresses, NULL was returned for NETWORK.
After my patch, this is no longer true, and we get a listen address for olddev
even if it is a type of NETWORK. So we have a real string on one side, the NULL
from user's XML on the other side and hence we think user wants to change the
listen address and we refuse it.
Therefore, we must take the type of listen address into account as well.
2013-06-20 16:27:35 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_NONE:
|
|
|
|
case VIR_DOMAIN_GRAPHICS_LISTEN_TYPE_LAST:
|
|
|
|
/* nada */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
conf: add <listen> subelement to domain <graphics> element
Once it's plugged in, the <listen> element will be an optional
replacement for the "listen" attribute that graphics elements already
have. If the <listen> element is type='address', it will have an
attribute called 'address' which will contain an IP address or dns
name that the guest's display server should listen on. If, however,
type='network', the <listen> element should have an attribute called
'network' that will be set to the name of a network configuration to
get the IP address from.
* docs/schemas/domain.rng: updated to allow the <listen> element
* docs/formatdomain.html.in: document the <listen> element and its
attributes.
* src/conf/domain_conf.[hc]:
1) The domain parser, formatter, and data structure are modified to
support 0 or more <listen> subelements to each <graphics>
element. The old style "legacy" listen attribute is also still
accepted, and will be stored internally just as if it were a
separate <listen> element. On output (i.e. format), the address
attribute of the first <listen> element of type 'address' will be
duplicated in the legacy "listen" attribute of the <graphic>
element.
2) The "listenAddr" attribute has been removed from the unions in
virDomainGRaphicsDef for graphics types vnc, rdp, and spice.
This attribute is now in the <listen> subelement (aka
virDomainGraphicsListenDef)
3) Helper functions were written to provide simple access
(both Get and Set) to the listen elements and their attributes.
* src/libvirt_private.syms: export the listen helper functions
* src/qemu/qemu_command.c, src/qemu/qemu_hotplug.c,
src/qemu/qemu_migration.c, src/vbox/vbox_tmpl.c,
src/vmx/vmx.c, src/xenxs/xen_sxpr.c, src/xenxs/xen_xm.c
Modify all these files to use the listen helper functions rather
than directly referencing the (now missing) listenAddr
attribute. There can be multiple <listen> elements to a single
<graphics>, but the drivers all currently only support one, so all
replacements of direct access with a helper function indicate index
"0".
* tests/* - only 3 of these are new files added explicitly to test the
new <listen> element. All the others have been modified to reflect
the fact that any legacy "listen" attributes passed in to the domain
parse will be saved in a <listen> element (i.e. one of the
virDomainGraphicsListenDefs), and during the domain format function,
both the <listen> element as well as the legacy attributes will be
output.
2011-07-07 04:20:28 +00:00
|
|
|
|
2010-12-16 16:10:54 +00:00
|
|
|
switch (dev->type) {
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_VNC:
|
2016-05-16 10:59:12 +00:00
|
|
|
if ((olddev->data.vnc.autoport != dev->data.vnc.autoport) ||
|
|
|
|
(!dev->data.vnc.autoport &&
|
|
|
|
(olddev->data.vnc.port != dev->data.vnc.port))) {
|
2014-12-08 13:48:31 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
2012-07-18 15:22:03 +00:00
|
|
|
_("cannot change port settings on vnc graphics"));
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
if (STRNEQ_NULLABLE(olddev->data.vnc.keymap, dev->data.vnc.keymap)) {
|
2014-12-08 13:48:31 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
2012-07-18 15:22:03 +00:00
|
|
|
_("cannot change keymap setting on vnc graphics"));
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2011-05-26 14:15:54 +00:00
|
|
|
/* If a password lifetime was, or is set, or action if connected has
|
|
|
|
* changed, then we must always run, even if new password matches
|
|
|
|
* old password */
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
if (olddev->data.vnc.auth.expires ||
|
|
|
|
dev->data.vnc.auth.expires ||
|
2011-05-26 14:15:54 +00:00
|
|
|
olddev->data.vnc.auth.connected != dev->data.vnc.auth.connected ||
|
2011-06-06 19:35:41 +00:00
|
|
|
STRNEQ_NULLABLE(olddev->data.vnc.auth.passwd,
|
|
|
|
dev->data.vnc.auth.passwd)) {
|
|
|
|
VIR_DEBUG("Updating password on VNC server %p %p",
|
2013-01-10 21:03:14 +00:00
|
|
|
dev->data.vnc.auth.passwd, cfg->vncPassword);
|
2011-06-06 19:35:41 +00:00
|
|
|
ret = qemuDomainChangeGraphicsPasswords(driver, vm,
|
|
|
|
VIR_DOMAIN_GRAPHICS_TYPE_VNC,
|
|
|
|
&dev->data.vnc.auth,
|
2014-08-12 02:54:42 +00:00
|
|
|
cfg->vncPassword,
|
|
|
|
QEMU_ASYNC_JOB_NONE);
|
2012-09-03 14:52:27 +00:00
|
|
|
if (ret < 0)
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
|
|
|
/* Steal the new dev's char * reference */
|
|
|
|
VIR_FREE(olddev->data.vnc.auth.passwd);
|
|
|
|
olddev->data.vnc.auth.passwd = dev->data.vnc.auth.passwd;
|
|
|
|
dev->data.vnc.auth.passwd = NULL;
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
olddev->data.vnc.auth.validTo = dev->data.vnc.auth.validTo;
|
|
|
|
olddev->data.vnc.auth.expires = dev->data.vnc.auth.expires;
|
2011-05-26 14:15:54 +00:00
|
|
|
olddev->data.vnc.auth.connected = dev->data.vnc.auth.connected;
|
2010-12-16 16:10:54 +00:00
|
|
|
} else {
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_SPICE:
|
2016-05-16 10:59:12 +00:00
|
|
|
if ((olddev->data.spice.autoport != dev->data.spice.autoport) ||
|
|
|
|
(!dev->data.spice.autoport &&
|
|
|
|
(olddev->data.spice.port != dev->data.spice.port)) ||
|
|
|
|
(!dev->data.spice.autoport &&
|
|
|
|
(olddev->data.spice.tlsPort != dev->data.spice.tlsPort))) {
|
2014-12-08 13:48:31 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
2012-07-18 15:22:03 +00:00
|
|
|
_("cannot change port settings on spice graphics"));
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
}
|
2011-06-06 19:35:41 +00:00
|
|
|
if (STRNEQ_NULLABLE(olddev->data.spice.keymap,
|
|
|
|
dev->data.spice.keymap)) {
|
2014-12-08 13:48:31 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
_("cannot change keymap setting on spice graphics"));
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
}
|
|
|
|
|
2012-06-14 12:48:42 +00:00
|
|
|
/* We must reset the password if it has changed but also if:
|
|
|
|
* - password lifetime is or was set
|
|
|
|
* - the requested action has changed
|
|
|
|
* - the action is "disconnect"
|
|
|
|
*/
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
if (olddev->data.spice.auth.expires ||
|
|
|
|
dev->data.spice.auth.expires ||
|
2011-05-26 14:15:54 +00:00
|
|
|
olddev->data.spice.auth.connected != dev->data.spice.auth.connected ||
|
2012-06-14 12:48:42 +00:00
|
|
|
dev->data.spice.auth.connected ==
|
|
|
|
VIR_DOMAIN_GRAPHICS_AUTH_CONNECTED_DISCONNECT ||
|
2011-06-06 19:35:41 +00:00
|
|
|
STRNEQ_NULLABLE(olddev->data.spice.auth.passwd,
|
|
|
|
dev->data.spice.auth.passwd)) {
|
|
|
|
VIR_DEBUG("Updating password on SPICE server %p %p",
|
2013-01-10 21:03:14 +00:00
|
|
|
dev->data.spice.auth.passwd, cfg->spicePassword);
|
2011-06-06 19:35:41 +00:00
|
|
|
ret = qemuDomainChangeGraphicsPasswords(driver, vm,
|
|
|
|
VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
|
|
|
|
&dev->data.spice.auth,
|
2014-08-12 02:54:42 +00:00
|
|
|
cfg->spicePassword,
|
|
|
|
QEMU_ASYNC_JOB_NONE);
|
2011-06-06 19:35:41 +00:00
|
|
|
|
2012-09-03 14:52:27 +00:00
|
|
|
if (ret < 0)
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2012-09-03 14:52:27 +00:00
|
|
|
|
2011-06-06 19:35:41 +00:00
|
|
|
/* Steal the new dev's char * reference */
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
VIR_FREE(olddev->data.spice.auth.passwd);
|
|
|
|
olddev->data.spice.auth.passwd = dev->data.spice.auth.passwd;
|
|
|
|
dev->data.spice.auth.passwd = NULL;
|
|
|
|
olddev->data.spice.auth.validTo = dev->data.spice.auth.validTo;
|
|
|
|
olddev->data.spice.auth.expires = dev->data.spice.auth.expires;
|
2011-05-26 14:15:54 +00:00
|
|
|
olddev->data.spice.auth.connected = dev->data.spice.auth.connected;
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
} else {
|
2011-05-09 09:24:09 +00:00
|
|
|
VIR_DEBUG("Not updating since password didn't change");
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
ret = 0;
|
|
|
|
}
|
2011-06-06 19:30:52 +00:00
|
|
|
break;
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
|
2018-02-14 09:43:59 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_SDL:
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_RDP:
|
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_DESKTOP:
|
2018-06-30 14:23:01 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_EGL_HEADLESS:
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2016-04-25 13:36:28 +00:00
|
|
|
_("unable to change config on '%s' graphics type"), type);
|
2010-12-16 16:10:54 +00:00
|
|
|
break;
|
2018-02-14 09:43:59 +00:00
|
|
|
case VIR_DOMAIN_GRAPHICS_TYPE_LAST:
|
|
|
|
default:
|
|
|
|
virReportEnumRangeError(virDomainGraphicsType, dev->type);
|
|
|
|
break;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2013-01-10 21:03:14 +00:00
|
|
|
virObjectUnref(cfg);
|
2010-12-16 16:10:54 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-09 06:59:16 +00:00
|
|
|
static int qemuComparePCIDevice(virDomainDefPtr def ATTRIBUTE_UNUSED,
|
2012-02-22 21:06:10 +00:00
|
|
|
virDomainDeviceDefPtr device ATTRIBUTE_UNUSED,
|
2012-02-23 17:59:21 +00:00
|
|
|
virDomainDeviceInfoPtr info1,
|
2011-05-09 06:59:16 +00:00
|
|
|
void *opaque)
|
|
|
|
{
|
2012-02-23 17:59:21 +00:00
|
|
|
virDomainDeviceInfoPtr info2 = opaque;
|
2011-05-09 06:59:16 +00:00
|
|
|
|
2012-02-23 17:59:21 +00:00
|
|
|
if (info1->type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI ||
|
|
|
|
info2->type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI)
|
2011-05-09 06:59:16 +00:00
|
|
|
return 0;
|
|
|
|
|
2013-04-03 16:09:47 +00:00
|
|
|
if (info1->addr.pci.domain == info2->addr.pci.domain &&
|
|
|
|
info1->addr.pci.bus == info2->addr.pci.bus &&
|
|
|
|
info1->addr.pci.slot == info2->addr.pci.slot &&
|
2012-02-23 17:59:21 +00:00
|
|
|
info1->addr.pci.function != info2->addr.pci.function)
|
2011-05-09 06:59:16 +00:00
|
|
|
return -1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool qemuIsMultiFunctionDevice(virDomainDefPtr def,
|
|
|
|
virDomainDeviceInfoPtr dev)
|
|
|
|
{
|
2017-10-18 13:33:45 +00:00
|
|
|
if (dev->type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI)
|
|
|
|
return false;
|
|
|
|
|
2011-05-09 06:59:16 +00:00
|
|
|
if (virDomainDeviceInfoIterate(def, qemuComparePCIDevice, dev) < 0)
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2014-05-27 10:09:09 +00:00
|
|
|
static int
|
2013-07-09 21:43:22 +00:00
|
|
|
qemuDomainRemoveDiskDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDiskDefPtr disk)
|
|
|
|
{
|
2018-07-11 09:16:41 +00:00
|
|
|
qemuHotplugDiskSourceDataPtr diskbackend = NULL;
|
2013-07-09 21:43:22 +00:00
|
|
|
virDomainDeviceDef dev;
|
2013-11-22 14:38:05 +00:00
|
|
|
virObjectEventPtr event;
|
2013-07-09 21:43:22 +00:00
|
|
|
size_t i;
|
2014-05-27 10:09:09 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2018-07-13 10:54:26 +00:00
|
|
|
int ret = -1;
|
2013-07-09 21:43:22 +00:00
|
|
|
|
|
|
|
VIR_DEBUG("Removing disk %s from domain %p %s",
|
|
|
|
disk->info.alias, vm, vm->def->name);
|
|
|
|
|
2018-10-04 15:49:00 +00:00
|
|
|
if (!(diskbackend = qemuHotplugDiskSourceRemovePrepare(disk, disk->src,
|
|
|
|
priv->qemuCaps)))
|
2014-05-27 10:09:09 +00:00
|
|
|
return -1;
|
|
|
|
|
2018-05-31 13:18:20 +00:00
|
|
|
for (i = 0; i < vm->def->ndisks; i++) {
|
|
|
|
if (vm->def->disks[i] == disk) {
|
|
|
|
virDomainDiskRemove(vm->def, i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-27 10:09:09 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2016-06-22 11:07:49 +00:00
|
|
|
|
2018-07-11 09:16:41 +00:00
|
|
|
qemuHotplugDiskSourceRemove(priv->mon, diskbackend);
|
qemu: Add TLS support for Veritas HyperScale (VxHS)
Alter qemu command line generation in order to possibly add TLS for
a suitably configured domain.
Sample TLS args generated by libvirt -
-object tls-creds-x509,id=objvirtio-disk0_tls0,dir=/etc/pki/qemu,\
endpoint=client,verify-peer=yes \
-drive file.driver=vxhs,file.tls-creds=objvirtio-disk0_tls0,\
file.vdisk-id=eb90327c-8302-4725-9e1b-4e85ed4dc251,\
file.server.type=tcp,file.server.host=192.168.0.1,\
file.server.port=9999,format=raw,if=none,\
id=drive-virtio-disk0,cache=none \
-device virtio-blk-pci,bus=pci.0,addr=0x4,drive=drive-virtio-disk0,\
id=virtio-disk0
Update the qemuxml2argvtest with a couple of examples. One for a
simple case and the other a bit more complex where multiple VxHS disks
are added where at least one uses a VxHS that doesn't require TLS
credentials and thus sets the domain disk source attribute "tls = 'no'".
Update the hotplug to be able to handle processing the tlsAlias whether
it's to add the TLS object when hotplugging a disk or to remove the TLS
object when hot unplugging a disk. The hot plug/unplug code is largely
generic, but the addition code does make the VXHS specific checks only
because it needs to grab the correct config directory and generate the
object as the command line would do.
Signed-off-by: Ashish Mittal <Ashish.Mittal@veritas.com>
Signed-off-by: John Ferlan <jferlan@redhat.com>
2017-08-30 15:06:00 +00:00
|
|
|
|
2014-12-16 14:50:20 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
2018-07-13 10:54:26 +00:00
|
|
|
goto cleanup;
|
2014-05-27 10:09:09 +00:00
|
|
|
|
2014-07-03 08:28:12 +00:00
|
|
|
virDomainAuditDisk(vm, disk->src, NULL, "detach", true);
|
2013-07-09 21:43:22 +00:00
|
|
|
|
2013-07-11 15:30:56 +00:00
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, disk->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2013-07-11 15:30:56 +00:00
|
|
|
|
2018-07-11 15:25:19 +00:00
|
|
|
qemuDomainReleaseDeviceAddress(vm, &disk->info, virDomainDiskGetSource(disk));
|
2013-07-09 21:43:22 +00:00
|
|
|
|
2018-07-11 15:21:36 +00:00
|
|
|
/* tear down disk security access */
|
2019-01-23 13:28:31 +00:00
|
|
|
qemuHotplugPrepareDiskSourceAccess(driver, vm, disk->src, true);
|
2016-11-15 15:53:04 +00:00
|
|
|
|
2013-07-09 21:43:22 +00:00
|
|
|
dev.type = VIR_DOMAIN_DEVICE_DISK;
|
|
|
|
dev.data.disk = disk;
|
|
|
|
ignore_value(qemuRemoveSharedDevice(driver, &dev, vm->def->name));
|
2017-10-20 11:24:49 +00:00
|
|
|
virDomainUSBAddressRelease(priv->usbaddrs, &disk->info);
|
2013-07-09 21:43:22 +00:00
|
|
|
|
2018-07-11 12:24:49 +00:00
|
|
|
if (qemuHotplugRemoveManagedPR(driver, vm, QEMU_ASYNC_JOB_NONE) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2018-07-13 10:54:26 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
2018-07-11 09:16:41 +00:00
|
|
|
qemuHotplugDiskSourceDataFree(diskbackend);
|
2013-07-09 21:43:22 +00:00
|
|
|
virDomainDiskDefFree(disk);
|
2018-07-13 10:54:26 +00:00
|
|
|
return ret;
|
2013-07-09 21:43:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-03 08:18:48 +00:00
|
|
|
static int
|
2013-07-11 15:30:56 +00:00
|
|
|
qemuDomainRemoveControllerDevice(virQEMUDriverPtr driver,
|
2013-07-09 22:10:32 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainControllerDefPtr controller)
|
|
|
|
{
|
2013-11-22 14:38:05 +00:00
|
|
|
virObjectEventPtr event;
|
2013-07-09 22:10:32 +00:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
VIR_DEBUG("Removing controller %s from domain %p %s",
|
|
|
|
controller->info.alias, vm, vm->def->name);
|
|
|
|
|
2013-07-11 15:30:56 +00:00
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, controller->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2013-07-11 15:30:56 +00:00
|
|
|
|
2013-07-09 22:10:32 +00:00
|
|
|
for (i = 0; i < vm->def->ncontrollers; i++) {
|
|
|
|
if (vm->def->controllers[i] == controller) {
|
|
|
|
virDomainControllerRemove(vm->def, i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &controller->info, NULL);
|
|
|
|
virDomainControllerDefFree(controller);
|
2014-06-03 08:18:48 +00:00
|
|
|
return 0;
|
2013-07-09 22:10:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-21 16:45:54 +00:00
|
|
|
static int
|
|
|
|
qemuDomainRemoveMemoryDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainMemoryDefPtr mem)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-06-15 13:34:04 +00:00
|
|
|
unsigned long long oldmem = virDomainDefGetMemoryTotal(vm->def);
|
2015-08-13 14:15:28 +00:00
|
|
|
unsigned long long newmem = oldmem - mem->size;
|
2015-01-21 16:45:54 +00:00
|
|
|
virObjectEventPtr event;
|
|
|
|
char *backendAlias = NULL;
|
|
|
|
int rc;
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
VIR_DEBUG("Removing memory device %s from domain %p %s",
|
|
|
|
mem->info.alias, vm, vm->def->name);
|
|
|
|
|
|
|
|
if (virAsprintf(&backendAlias, "mem%s", mem->info.alias) < 0)
|
2015-08-13 14:15:28 +00:00
|
|
|
return -1;
|
2015-01-21 16:45:54 +00:00
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuMonitorDelObject(priv->mon, backendAlias);
|
2015-08-13 14:15:28 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
rc = -1;
|
|
|
|
|
|
|
|
VIR_FREE(backendAlias);
|
|
|
|
|
|
|
|
virDomainAuditMemory(vm, oldmem, newmem, "update", rc == 0);
|
|
|
|
if (rc < 0)
|
|
|
|
return -1;
|
2015-01-21 16:45:54 +00:00
|
|
|
|
2017-01-20 13:24:35 +00:00
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, mem->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2017-01-20 13:24:35 +00:00
|
|
|
|
2015-01-21 16:45:54 +00:00
|
|
|
if ((idx = virDomainMemoryFindByDef(vm->def, mem)) >= 0)
|
|
|
|
virDomainMemoryRemove(vm->def, idx);
|
|
|
|
|
2016-08-04 13:26:09 +00:00
|
|
|
if (qemuSecurityRestoreMemoryLabel(driver, vm, mem) < 0)
|
|
|
|
VIR_WARN("Unable to restore security label on memdev");
|
|
|
|
|
2017-02-22 15:33:12 +00:00
|
|
|
if (qemuTeardownMemoryDevicesCgroup(vm, mem) < 0)
|
|
|
|
VIR_WARN("Unable to remove memory device cgroup ACL");
|
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceTeardownMemory(vm, mem) < 0)
|
2017-02-22 16:37:39 +00:00
|
|
|
VIR_WARN("Unable to remove memory device from /dev");
|
|
|
|
|
2018-01-11 12:02:52 +00:00
|
|
|
if (qemuProcessDestroyMemoryBackingPath(driver, vm, mem) < 0)
|
|
|
|
VIR_WARN("Unable to destroy memory backing path");
|
|
|
|
|
2015-01-21 16:45:54 +00:00
|
|
|
virDomainMemoryDefFree(mem);
|
2015-11-06 15:39:31 +00:00
|
|
|
|
2016-04-06 13:57:57 +00:00
|
|
|
/* fix the balloon size */
|
|
|
|
ignore_value(qemuProcessRefreshBalloonState(driver, vm, QEMU_ASYNC_JOB_NONE));
|
|
|
|
|
2015-11-06 15:39:31 +00:00
|
|
|
/* decrease the mlock limit after memory unplug if necessary */
|
2015-11-23 16:57:40 +00:00
|
|
|
ignore_value(qemuDomainAdjustMaxMemLock(vm));
|
2015-11-06 15:39:31 +00:00
|
|
|
|
2015-08-13 14:15:28 +00:00
|
|
|
return 0;
|
2015-01-21 16:45:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-11 09:14:16 +00:00
|
|
|
static void
|
|
|
|
qemuDomainRemovePCIHostDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
2015-10-20 12:12:48 +00:00
|
|
|
qemuHostdevReAttachPCIDevices(driver, vm->def->name, &hostdev, 1);
|
2013-07-11 09:14:16 +00:00
|
|
|
qemuDomainReleaseDeviceAddress(vm, hostdev->info, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemuDomainRemoveUSBHostDevice(virQEMUDriverPtr driver,
|
2014-03-05 06:33:30 +00:00
|
|
|
virDomainObjPtr vm,
|
2013-07-11 09:14:16 +00:00
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
2015-10-20 12:12:48 +00:00
|
|
|
qemuHostdevReAttachUSBDevices(driver, vm->def->name, &hostdev, 1);
|
2015-08-12 14:52:19 +00:00
|
|
|
qemuDomainReleaseDeviceAddress(vm, hostdev->info, NULL);
|
2013-07-11 09:14:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemuDomainRemoveSCSIHostDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
2015-10-20 12:12:48 +00:00
|
|
|
qemuHostdevReAttachSCSIDevices(driver, vm->def->name, &hostdev, 1);
|
2013-07-11 09:14:16 +00:00
|
|
|
}
|
|
|
|
|
2016-11-22 03:58:19 +00:00
|
|
|
static void
|
|
|
|
qemuDomainRemoveSCSIVHostDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
|
|
|
qemuHostdevReAttachSCSIVHostDevices(driver, vm->def->name, &hostdev, 1);
|
|
|
|
}
|
|
|
|
|
2018-03-26 08:06:07 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
qemuDomainRemoveMediatedDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
|
|
|
qemuHostdevReAttachMediatedDevices(driver, vm->def->name, &hostdev, 1);
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, hostdev->info, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-03 08:18:48 +00:00
|
|
|
static int
|
2013-07-11 09:14:16 +00:00
|
|
|
qemuDomainRemoveHostDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainHostdevDefPtr hostdev)
|
|
|
|
{
|
2013-10-18 08:39:08 +00:00
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2013-07-11 09:14:16 +00:00
|
|
|
virDomainNetDefPtr net = NULL;
|
2013-11-22 14:38:05 +00:00
|
|
|
virObjectEventPtr event;
|
2013-07-11 09:14:16 +00:00
|
|
|
size_t i;
|
2014-09-23 22:53:25 +00:00
|
|
|
int ret = -1;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-07-18 17:22:29 +00:00
|
|
|
char *drivealias = NULL;
|
2017-09-15 17:17:59 +00:00
|
|
|
char *objAlias = NULL;
|
2016-06-17 11:14:56 +00:00
|
|
|
bool is_vfio = false;
|
2013-07-11 09:14:16 +00:00
|
|
|
|
|
|
|
VIR_DEBUG("Removing host device %s from domain %p %s",
|
|
|
|
hostdev->info->alias, vm, vm->def->name);
|
|
|
|
|
2016-06-17 11:14:56 +00:00
|
|
|
if (hostdev->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI) {
|
|
|
|
int backend = hostdev->source.subsys.u.pci.backend;
|
|
|
|
is_vfio = backend == VIR_DOMAIN_HOSTDEV_PCI_BACKEND_VFIO;
|
|
|
|
}
|
|
|
|
|
2014-09-23 22:53:25 +00:00
|
|
|
if (hostdev->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI) {
|
2017-09-15 17:17:59 +00:00
|
|
|
virDomainHostdevSubsysSCSIPtr scsisrc = &hostdev->source.subsys.u.scsi;
|
|
|
|
virDomainHostdevSubsysSCSIiSCSIPtr iscsisrc = &scsisrc->u.iscsi;
|
|
|
|
|
2016-07-18 17:22:29 +00:00
|
|
|
if (!(drivealias = qemuAliasFromHostdev(hostdev)))
|
2014-09-23 22:53:25 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2017-09-15 17:17:59 +00:00
|
|
|
/* Look for the markers that the iSCSI hostdev was added with a
|
|
|
|
* secret object to manage the username/password. If present, let's
|
|
|
|
* attempt to remove the object as well. */
|
|
|
|
if (scsisrc->protocol == VIR_DOMAIN_HOSTDEV_SCSI_PROTOCOL_TYPE_ISCSI &&
|
|
|
|
virQEMUCapsGet(priv->qemuCaps, QEMU_CAPS_ISCSI_PASSWORD_SECRET) &&
|
2018-05-22 07:18:34 +00:00
|
|
|
qemuDomainStorageSourceHasAuth(iscsisrc->src)) {
|
2017-09-15 17:17:59 +00:00
|
|
|
if (!(objAlias = qemuDomainGetSecretAESAlias(hostdev->info->alias, false)))
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2014-09-23 22:53:25 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2016-07-18 17:22:29 +00:00
|
|
|
qemuMonitorDriveDel(priv->mon, drivealias);
|
2017-09-15 17:17:59 +00:00
|
|
|
|
|
|
|
/* If it fails, then so be it - it was a best shot */
|
|
|
|
if (objAlias)
|
|
|
|
ignore_value(qemuMonitorDelObject(priv->mon, objAlias));
|
|
|
|
|
2014-12-16 14:50:20 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
2014-09-23 22:53:25 +00:00
|
|
|
}
|
|
|
|
|
2013-07-11 15:30:56 +00:00
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, hostdev->info->alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2013-07-11 15:30:56 +00:00
|
|
|
|
2013-07-11 09:14:16 +00:00
|
|
|
if (hostdev->parent.type == VIR_DOMAIN_DEVICE_NET) {
|
|
|
|
net = hostdev->parent.data.net;
|
|
|
|
|
|
|
|
for (i = 0; i < vm->def->nnets; i++) {
|
|
|
|
if (vm->def->nets[i] == net) {
|
|
|
|
virDomainNetRemove(vm->def, i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < vm->def->nhostdevs; i++) {
|
|
|
|
if (vm->def->hostdevs[i] == hostdev) {
|
|
|
|
virDomainHostdevRemove(vm->def, i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainAuditHostdev(vm, hostdev, "detach", true);
|
|
|
|
|
2016-06-17 11:14:56 +00:00
|
|
|
if (!is_vfio &&
|
2016-11-16 14:27:47 +00:00
|
|
|
qemuSecurityRestoreHostdevLabel(driver, vm, hostdev) < 0)
|
2016-06-17 11:14:56 +00:00
|
|
|
VIR_WARN("Failed to restore host device labelling");
|
2016-03-28 11:34:03 +00:00
|
|
|
|
2016-03-28 11:40:57 +00:00
|
|
|
if (qemuTeardownHostdevCgroup(vm, hostdev) < 0)
|
|
|
|
VIR_WARN("Failed to remove host device cgroup ACL");
|
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceTeardownHostdev(vm, hostdev) < 0)
|
2016-11-16 14:27:47 +00:00
|
|
|
VIR_WARN("Unable to remove host device from /dev");
|
|
|
|
|
2018-04-25 12:42:34 +00:00
|
|
|
switch ((virDomainHostdevSubsysType)hostdev->source.subsys.type) {
|
2013-07-11 09:14:16 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI:
|
|
|
|
qemuDomainRemovePCIHostDevice(driver, vm, hostdev);
|
2015-11-18 12:56:09 +00:00
|
|
|
/* QEMU might no longer need to lock as much memory, eg. we just
|
|
|
|
* detached the last VFIO device, so adjust the limit here */
|
|
|
|
if (qemuDomainAdjustMaxMemLock(vm) < 0)
|
|
|
|
VIR_WARN("Failed to adjust locked memory limit");
|
2013-07-11 09:14:16 +00:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB:
|
|
|
|
qemuDomainRemoveUSBHostDevice(driver, vm, hostdev);
|
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI:
|
|
|
|
qemuDomainRemoveSCSIHostDevice(driver, vm, hostdev);
|
|
|
|
break;
|
2016-11-22 03:58:16 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI_HOST:
|
2016-11-22 03:58:19 +00:00
|
|
|
qemuDomainRemoveSCSIVHostDevice(driver, vm, hostdev);
|
2016-11-22 03:58:16 +00:00
|
|
|
break;
|
2017-01-31 16:26:36 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_MDEV:
|
2018-03-26 08:06:07 +00:00
|
|
|
qemuDomainRemoveMediatedDevice(driver, vm, hostdev);
|
2017-01-31 16:26:36 +00:00
|
|
|
break;
|
2013-07-11 09:14:16 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_LAST:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainHostdevDefFree(hostdev);
|
|
|
|
|
|
|
|
if (net) {
|
2018-01-25 09:35:47 +00:00
|
|
|
virDomainNetReleaseActualDevice(vm->def, net);
|
2013-07-11 09:14:16 +00:00
|
|
|
virDomainNetDefFree(net);
|
|
|
|
}
|
2015-11-18 12:56:09 +00:00
|
|
|
|
2014-09-23 22:53:25 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
2016-07-18 17:22:29 +00:00
|
|
|
VIR_FREE(drivealias);
|
2017-09-15 17:17:59 +00:00
|
|
|
VIR_FREE(objAlias);
|
2013-10-18 08:39:08 +00:00
|
|
|
virObjectUnref(cfg);
|
2014-09-23 22:53:25 +00:00
|
|
|
return ret;
|
2013-07-11 09:14:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-05-27 09:50:41 +00:00
|
|
|
static int
|
2013-10-18 09:28:40 +00:00
|
|
|
qemuDomainRemoveNetDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainNetDefPtr net)
|
|
|
|
{
|
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
2014-05-27 09:50:41 +00:00
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2013-11-22 14:38:05 +00:00
|
|
|
virObjectEventPtr event;
|
2014-05-27 09:50:41 +00:00
|
|
|
char *hostnet_name = NULL;
|
2016-08-15 16:01:55 +00:00
|
|
|
char *charDevAlias = NULL;
|
2013-10-18 09:28:40 +00:00
|
|
|
size_t i;
|
2014-05-27 09:50:41 +00:00
|
|
|
int ret = -1;
|
2016-08-15 16:01:55 +00:00
|
|
|
int actualType = virDomainNetGetActualType(net);
|
2013-10-18 09:28:40 +00:00
|
|
|
|
2016-08-15 16:01:55 +00:00
|
|
|
if (actualType == VIR_DOMAIN_NET_TYPE_HOSTDEV) {
|
2013-10-18 09:34:53 +00:00
|
|
|
/* this function handles all hostdev and netdev cleanup */
|
2014-06-03 08:18:48 +00:00
|
|
|
ret = qemuDomainRemoveHostDevice(driver, vm,
|
|
|
|
virDomainNetGetActualHostdev(net));
|
2014-05-27 09:49:53 +00:00
|
|
|
goto cleanup;
|
2013-10-18 09:34:53 +00:00
|
|
|
}
|
|
|
|
|
2013-10-18 09:28:40 +00:00
|
|
|
VIR_DEBUG("Removing network interface %s from domain %p %s",
|
|
|
|
net->info.alias, vm, vm->def->name);
|
|
|
|
|
2016-08-15 16:01:55 +00:00
|
|
|
if (virAsprintf(&hostnet_name, "host%s", net->info.alias) < 0 ||
|
2016-10-18 14:37:23 +00:00
|
|
|
!(charDevAlias = qemuAliasChardevFromDevAlias(net->info.alias)))
|
2014-05-27 09:50:41 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2016-08-15 16:01:55 +00:00
|
|
|
|
2014-05-27 09:50:41 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2018-03-28 21:36:13 +00:00
|
|
|
if (qemuMonitorRemoveNetdev(priv->mon, hostnet_name) < 0) {
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
2014-05-27 09:50:41 +00:00
|
|
|
goto cleanup;
|
2018-03-28 21:36:13 +00:00
|
|
|
virDomainAuditNet(vm, net, NULL, "detach", false);
|
|
|
|
goto cleanup;
|
2014-05-27 09:50:41 +00:00
|
|
|
}
|
2016-08-15 16:01:55 +00:00
|
|
|
|
|
|
|
if (actualType == VIR_DOMAIN_NET_TYPE_VHOSTUSER) {
|
|
|
|
/* vhostuser has a chardev too */
|
|
|
|
if (qemuMonitorDetachCharDev(priv->mon, charDevAlias) < 0) {
|
|
|
|
/* well, this is a messy situation. Guest visible PCI device has
|
|
|
|
* been removed, netdev too but chardev not. The best seems to be
|
|
|
|
* to just ignore the error and carry on.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-16 14:50:20 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
2014-05-27 09:50:41 +00:00
|
|
|
|
2013-10-18 09:28:40 +00:00
|
|
|
virDomainAuditNet(vm, net, NULL, "detach", true);
|
|
|
|
|
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, net->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2013-10-18 09:28:40 +00:00
|
|
|
|
|
|
|
for (i = 0; i < vm->def->nnets; i++) {
|
|
|
|
if (vm->def->nets[i] == net) {
|
|
|
|
virDomainNetRemove(vm->def, i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &net->info, NULL);
|
|
|
|
virDomainConfNWFilterTeardown(net);
|
|
|
|
|
2014-09-30 10:55:23 +00:00
|
|
|
if (cfg->macFilter && (net->ifname != NULL)) {
|
|
|
|
ignore_value(ebtablesRemoveForwardAllowIn(driver->ebtables,
|
|
|
|
net->ifname,
|
|
|
|
&net->mac));
|
|
|
|
}
|
|
|
|
|
2016-08-15 16:01:55 +00:00
|
|
|
if (actualType == VIR_DOMAIN_NET_TYPE_DIRECT) {
|
2013-10-18 09:28:40 +00:00
|
|
|
ignore_value(virNetDevMacVLanDeleteWithVPortProfile(
|
|
|
|
net->ifname, &net->mac,
|
|
|
|
virDomainNetGetActualDirectDev(net),
|
|
|
|
virDomainNetGetActualDirectMode(net),
|
|
|
|
virDomainNetGetActualVirtPortProfile(net),
|
|
|
|
cfg->stateDir));
|
|
|
|
}
|
|
|
|
|
2017-06-28 10:56:47 +00:00
|
|
|
qemuDomainNetDeviceVportRemove(net);
|
2013-10-18 09:28:40 +00:00
|
|
|
|
2018-01-25 09:35:47 +00:00
|
|
|
virDomainNetReleaseActualDevice(vm->def, net);
|
2013-10-18 09:28:40 +00:00
|
|
|
virDomainNetDefFree(net);
|
2014-05-27 09:50:41 +00:00
|
|
|
ret = 0;
|
2014-05-27 09:49:53 +00:00
|
|
|
|
|
|
|
cleanup:
|
2013-10-18 09:28:40 +00:00
|
|
|
virObjectUnref(cfg);
|
2016-08-15 16:01:55 +00:00
|
|
|
VIR_FREE(charDevAlias);
|
2014-05-27 09:50:41 +00:00
|
|
|
VIR_FREE(hostnet_name);
|
|
|
|
return ret;
|
2013-10-18 09:28:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-05-27 11:30:04 +00:00
|
|
|
static int
|
2013-07-11 15:30:56 +00:00
|
|
|
qemuDomainRemoveChrDevice(virQEMUDriverPtr driver,
|
2013-07-16 19:16:09 +00:00
|
|
|
virDomainObjPtr vm,
|
2019-02-11 13:16:58 +00:00
|
|
|
virDomainChrDefPtr chr,
|
|
|
|
bool monitor)
|
2013-07-16 19:16:09 +00:00
|
|
|
{
|
2013-11-22 14:38:05 +00:00
|
|
|
virObjectEventPtr event;
|
2014-05-27 11:30:04 +00:00
|
|
|
char *charAlias = NULL;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
int ret = -1;
|
2019-02-11 13:16:58 +00:00
|
|
|
int rc = 0;
|
2013-07-11 15:30:56 +00:00
|
|
|
|
2013-07-16 19:16:09 +00:00
|
|
|
VIR_DEBUG("Removing character device %s from domain %p %s",
|
|
|
|
chr->info.alias, vm, vm->def->name);
|
|
|
|
|
2019-02-14 13:13:08 +00:00
|
|
|
if (!(charAlias = qemuAliasChardevFromDevAlias(chr->info.alias)))
|
|
|
|
goto cleanup;
|
2014-05-27 11:30:04 +00:00
|
|
|
|
2019-02-14 13:13:08 +00:00
|
|
|
if (monitor) {
|
2019-02-11 13:16:58 +00:00
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuMonitorDetachCharDev(priv->mon, charAlias);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2014-05-27 11:30:04 +00:00
|
|
|
|
2017-12-20 11:36:26 +00:00
|
|
|
if (rc == 0 &&
|
|
|
|
qemuDomainDelChardevTLSObjects(driver, vm, chr->source, charAlias) < 0)
|
2017-12-19 22:46:41 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2014-07-03 08:59:58 +00:00
|
|
|
virDomainAuditChardev(vm, chr, NULL, "detach", rc == 0);
|
|
|
|
|
|
|
|
if (rc < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-18 10:45:44 +00:00
|
|
|
if (qemuTeardownChardevCgroup(vm, chr) < 0)
|
|
|
|
VIR_WARN("Failed to remove chr device cgroup ACL");
|
|
|
|
|
2017-12-01 12:10:35 +00:00
|
|
|
if (qemuSecurityRestoreChardevLabel(driver, vm, chr) < 0)
|
|
|
|
VIR_WARN("Unable to restore security label on char device");
|
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceTeardownChardev(vm, chr) < 0)
|
2016-11-18 13:53:27 +00:00
|
|
|
VIR_WARN("Unable to remove chr device from /dev");
|
|
|
|
|
2013-07-11 15:30:56 +00:00
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, chr->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2013-07-11 15:30:56 +00:00
|
|
|
|
2018-05-23 16:32:26 +00:00
|
|
|
qemuDomainReleaseDeviceAddress(vm, &chr->info, NULL);
|
2013-07-16 19:16:09 +00:00
|
|
|
qemuDomainChrRemove(vm->def, chr);
|
|
|
|
virDomainChrDefFree(chr);
|
2014-05-27 11:30:04 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(charAlias);
|
|
|
|
return ret;
|
2013-07-16 19:16:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-17 05:09:38 +00:00
|
|
|
static int
|
|
|
|
qemuDomainRemoveRNGDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainRNGDefPtr rng)
|
|
|
|
{
|
|
|
|
virObjectEventPtr event;
|
|
|
|
char *charAlias = NULL;
|
|
|
|
char *objAlias = NULL;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
ssize_t idx;
|
|
|
|
int ret = -1;
|
2018-12-04 13:30:37 +00:00
|
|
|
int rc = 0;
|
2015-01-17 05:09:38 +00:00
|
|
|
|
|
|
|
VIR_DEBUG("Removing RNG device %s from domain %p %s",
|
|
|
|
rng->info.alias, vm, vm->def->name);
|
|
|
|
|
2016-10-24 19:59:11 +00:00
|
|
|
|
2015-01-17 05:09:38 +00:00
|
|
|
if (virAsprintf(&objAlias, "obj%s", rng->info.alias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-10-18 14:37:23 +00:00
|
|
|
if (!(charAlias = qemuAliasChardevFromDevAlias(rng->info.alias)))
|
2015-01-17 05:09:38 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
2016-10-24 19:59:11 +00:00
|
|
|
|
2018-12-04 13:30:37 +00:00
|
|
|
if (qemuDomainDetachExtensionDevice(priv->mon, &rng->info) < 0)
|
|
|
|
rc = -1;
|
|
|
|
|
|
|
|
if (rc == 0 &&
|
|
|
|
qemuMonitorDelObject(priv->mon, objAlias) < 0)
|
|
|
|
rc = -1;
|
|
|
|
|
|
|
|
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD &&
|
|
|
|
rc == 0 &&
|
|
|
|
qemuMonitorDetachCharDev(priv->mon, charAlias) < 0)
|
|
|
|
rc = -1;
|
2015-01-17 05:09:38 +00:00
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2017-12-19 22:46:41 +00:00
|
|
|
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD &&
|
|
|
|
rc == 0 &&
|
2017-12-20 11:36:26 +00:00
|
|
|
qemuDomainDelChardevTLSObjects(driver, vm, rng->source.chardev,
|
|
|
|
charAlias) < 0)
|
2018-12-04 13:30:37 +00:00
|
|
|
rc = -1;
|
2017-12-19 22:46:41 +00:00
|
|
|
|
2015-01-17 05:09:38 +00:00
|
|
|
virDomainAuditRNG(vm, rng, NULL, "detach", rc == 0);
|
|
|
|
|
|
|
|
if (rc < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-18 10:17:51 +00:00
|
|
|
if (qemuTeardownRNGCgroup(vm, rng) < 0)
|
|
|
|
VIR_WARN("Failed to remove RNG device cgroup ACL");
|
|
|
|
|
2017-11-24 16:52:15 +00:00
|
|
|
if (qemuDomainNamespaceTeardownRNG(vm, rng) < 0)
|
2016-11-18 14:19:12 +00:00
|
|
|
VIR_WARN("Unable to remove RNG device from /dev");
|
|
|
|
|
2015-07-07 13:33:53 +00:00
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, rng->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2015-01-17 05:09:38 +00:00
|
|
|
|
|
|
|
if ((idx = virDomainRNGFind(vm->def, rng)) >= 0)
|
|
|
|
virDomainRNGRemove(vm->def, idx);
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &rng->info, NULL);
|
|
|
|
virDomainRNGDefFree(rng);
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(charAlias);
|
|
|
|
VIR_FREE(objAlias);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-12 13:40:48 +00:00
|
|
|
static int
|
|
|
|
qemuDomainRemoveShmemDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainShmemDefPtr shmem)
|
|
|
|
{
|
|
|
|
int rc;
|
|
|
|
int ret = -1;
|
|
|
|
ssize_t idx = -1;
|
|
|
|
char *charAlias = NULL;
|
|
|
|
char *memAlias = NULL;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virObjectEventPtr event = NULL;
|
|
|
|
|
|
|
|
VIR_DEBUG("Removing shmem device %s from domain %p %s",
|
|
|
|
shmem->info.alias, vm, vm->def->name);
|
|
|
|
|
|
|
|
if (shmem->server.enabled) {
|
|
|
|
if (virAsprintf(&charAlias, "char%s", shmem->info.alias) < 0)
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
if (virAsprintf(&memAlias, "shmmem-%s", shmem->info.alias) < 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
|
|
|
if (shmem->server.enabled)
|
|
|
|
rc = qemuMonitorDetachCharDev(priv->mon, charAlias);
|
|
|
|
else
|
|
|
|
rc = qemuMonitorDelObject(priv->mon, memAlias);
|
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
virDomainAuditShmem(vm, shmem, "detach", rc == 0);
|
|
|
|
|
|
|
|
if (rc < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, shmem->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2016-09-12 13:40:48 +00:00
|
|
|
|
|
|
|
if ((idx = virDomainShmemDefFind(vm->def, shmem)) >= 0)
|
|
|
|
virDomainShmemDefRemove(vm->def, idx);
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &shmem->info, NULL);
|
|
|
|
virDomainShmemDefFree(shmem);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(charAlias);
|
|
|
|
VIR_FREE(memAlias);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-05 09:08:36 +00:00
|
|
|
static int
|
|
|
|
qemuDomainRemoveWatchdog(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainWatchdogDefPtr watchdog)
|
|
|
|
{
|
|
|
|
virObjectEventPtr event = NULL;
|
|
|
|
|
|
|
|
VIR_DEBUG("Removing watchdog %s from domain %p %s",
|
|
|
|
watchdog->info.alias, vm, vm->def->name);
|
|
|
|
|
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, watchdog->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2017-09-05 09:08:36 +00:00
|
|
|
qemuDomainReleaseDeviceAddress(vm, &watchdog->info, NULL);
|
|
|
|
virDomainWatchdogDefFree(vm->def->watchdog);
|
|
|
|
vm->def->watchdog = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-10-17 14:44:24 +00:00
|
|
|
static int
|
|
|
|
qemuDomainRemoveInputDevice(virDomainObjPtr vm,
|
|
|
|
virDomainInputDefPtr dev)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virQEMUDriverPtr driver = priv->driver;
|
|
|
|
virObjectEventPtr event = NULL;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
VIR_DEBUG("Removing input device %s from domain %p %s",
|
|
|
|
dev->info.alias, vm, vm->def->name);
|
|
|
|
|
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, dev->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2017-10-17 14:44:24 +00:00
|
|
|
for (i = 0; i < vm->def->ninputs; i++) {
|
|
|
|
if (vm->def->inputs[i] == dev)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &dev->info, NULL);
|
2017-11-21 12:56:37 +00:00
|
|
|
if (qemuSecurityRestoreInputLabel(vm, dev) < 0)
|
|
|
|
VIR_WARN("Unable to restore security label on input device");
|
|
|
|
|
|
|
|
if (qemuTeardownInputCgroup(vm, dev) < 0)
|
|
|
|
VIR_WARN("Unable to remove input device cgroup ACL");
|
|
|
|
|
|
|
|
if (qemuDomainNamespaceTeardownInput(vm, dev) < 0)
|
|
|
|
VIR_WARN("Unable to remove input device from /dev");
|
|
|
|
|
2017-10-17 14:44:24 +00:00
|
|
|
virDomainInputDefFree(vm->def->inputs[i]);
|
|
|
|
VIR_DELETE_ELEMENT(vm->def->inputs, i, vm->def->ninputs);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-30 10:49:04 +00:00
|
|
|
static int
|
|
|
|
qemuDomainRemoveVsockDevice(virDomainObjPtr vm,
|
|
|
|
virDomainVsockDefPtr dev)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virQEMUDriverPtr driver = priv->driver;
|
|
|
|
virObjectEventPtr event = NULL;
|
|
|
|
|
|
|
|
VIR_DEBUG("Removing vsock device %s from domain %p %s",
|
|
|
|
dev->info.alias, vm, vm->def->name);
|
|
|
|
|
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, dev->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2018-05-30 10:49:04 +00:00
|
|
|
qemuDomainReleaseDeviceAddress(vm, &dev->info, NULL);
|
|
|
|
virDomainVsockDefFree(vm->def->vsock);
|
|
|
|
vm->def->vsock = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-05 02:47:47 +00:00
|
|
|
static int
|
|
|
|
qemuDomainRemoveRedirdevDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainRedirdevDefPtr dev)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virObjectEventPtr event;
|
|
|
|
char *charAlias = NULL;
|
|
|
|
ssize_t idx;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
VIR_DEBUG("Removing redirdev device %s from domain %p %s",
|
|
|
|
dev->info.alias, vm, vm->def->name);
|
|
|
|
|
|
|
|
if (!(charAlias = qemuAliasChardevFromDevAlias(dev->info.alias)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
/* DeviceDel from Detach may remove chardev,
|
|
|
|
* so we cannot rely on return status to delete TLS chardevs.
|
|
|
|
*/
|
|
|
|
ignore_value(qemuMonitorDetachCharDev(priv->mon, charAlias));
|
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuDomainDelChardevTLSObjects(driver, vm, dev->source, charAlias) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
virDomainAuditRedirdev(vm, dev, "detach", true);
|
|
|
|
|
|
|
|
event = virDomainEventDeviceRemovedNewFromObj(vm, dev->info.alias);
|
2018-06-12 17:33:02 +00:00
|
|
|
virObjectEventStateQueue(driver->domainEventState, event);
|
2018-01-05 02:47:47 +00:00
|
|
|
|
|
|
|
if ((idx = virDomainRedirdevDefFind(vm->def, dev)) >= 0)
|
|
|
|
virDomainRedirdevDefRemove(vm->def, idx);
|
|
|
|
qemuDomainReleaseDeviceAddress(vm, &dev->info, NULL);
|
|
|
|
virDomainRedirdevDefFree(dev);
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(charAlias);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-12-16 14:50:20 +00:00
|
|
|
int
|
2013-07-11 15:11:02 +00:00
|
|
|
qemuDomainRemoveDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDeviceDefPtr dev)
|
|
|
|
{
|
2014-12-16 14:50:20 +00:00
|
|
|
int ret = -1;
|
2018-04-25 12:42:34 +00:00
|
|
|
switch ((virDomainDeviceType)dev->type) {
|
2013-07-11 15:11:02 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_DISK:
|
2014-12-16 14:50:20 +00:00
|
|
|
ret = qemuDomainRemoveDiskDevice(driver, vm, dev->data.disk);
|
2013-07-11 15:11:02 +00:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_DEVICE_CONTROLLER:
|
2014-12-16 14:50:20 +00:00
|
|
|
ret = qemuDomainRemoveControllerDevice(driver, vm, dev->data.controller);
|
2013-07-11 15:11:02 +00:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_DEVICE_NET:
|
2014-12-16 14:50:20 +00:00
|
|
|
ret = qemuDomainRemoveNetDevice(driver, vm, dev->data.net);
|
2013-07-11 15:11:02 +00:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_DEVICE_HOSTDEV:
|
2014-12-16 14:50:20 +00:00
|
|
|
ret = qemuDomainRemoveHostDevice(driver, vm, dev->data.hostdev);
|
2013-07-11 15:11:02 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_DEVICE_CHR:
|
2019-02-11 13:16:58 +00:00
|
|
|
ret = qemuDomainRemoveChrDevice(driver, vm, dev->data.chr, true);
|
2013-07-11 15:11:02 +00:00
|
|
|
break;
|
2015-01-17 05:09:38 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_RNG:
|
2015-05-31 13:27:25 +00:00
|
|
|
ret = qemuDomainRemoveRNGDevice(driver, vm, dev->data.rng);
|
2015-01-17 05:09:38 +00:00
|
|
|
break;
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2014-09-29 17:02:04 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_MEMORY:
|
2015-01-21 16:45:54 +00:00
|
|
|
ret = qemuDomainRemoveMemoryDevice(driver, vm, dev->data.memory);
|
|
|
|
break;
|
2014-09-29 17:02:04 +00:00
|
|
|
|
2016-09-12 13:40:48 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_SHMEM:
|
|
|
|
ret = qemuDomainRemoveShmemDevice(driver, vm, dev->data.shmem);
|
|
|
|
break;
|
|
|
|
|
2017-12-14 09:45:31 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_INPUT:
|
|
|
|
ret = qemuDomainRemoveInputDevice(vm, dev->data.input);
|
|
|
|
break;
|
|
|
|
|
2018-01-05 02:47:47 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_REDIRDEV:
|
|
|
|
ret = qemuDomainRemoveRedirdevDevice(driver, vm, dev->data.redirdev);
|
|
|
|
break;
|
|
|
|
|
2018-03-31 08:12:17 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_WATCHDOG:
|
|
|
|
ret = qemuDomainRemoveWatchdog(driver, vm, dev->data.watchdog);
|
|
|
|
break;
|
|
|
|
|
2018-05-30 10:49:04 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_VSOCK:
|
|
|
|
ret = qemuDomainRemoveVsockDevice(vm, dev->data.vsock);
|
|
|
|
break;
|
|
|
|
|
2013-07-11 15:11:02 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_NONE:
|
|
|
|
case VIR_DOMAIN_DEVICE_LEASE:
|
|
|
|
case VIR_DOMAIN_DEVICE_FS:
|
|
|
|
case VIR_DOMAIN_DEVICE_SOUND:
|
|
|
|
case VIR_DOMAIN_DEVICE_VIDEO:
|
|
|
|
case VIR_DOMAIN_DEVICE_GRAPHICS:
|
|
|
|
case VIR_DOMAIN_DEVICE_HUB:
|
|
|
|
case VIR_DOMAIN_DEVICE_SMARTCARD:
|
|
|
|
case VIR_DOMAIN_DEVICE_MEMBALLOON:
|
|
|
|
case VIR_DOMAIN_DEVICE_NVRAM:
|
2014-11-30 19:54:50 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_TPM:
|
2014-11-30 19:54:50 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_PANIC:
|
2016-06-22 14:28:22 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_IOMMU:
|
2013-07-11 15:11:02 +00:00
|
|
|
case VIR_DOMAIN_DEVICE_LAST:
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("don't know how to remove a %s device"),
|
|
|
|
virDomainDeviceTypeToString(dev->type));
|
|
|
|
break;
|
|
|
|
}
|
2014-12-16 14:50:20 +00:00
|
|
|
return ret;
|
2013-07-11 15:11:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
2016-08-04 21:53:13 +00:00
|
|
|
qemuDomainMarkDeviceAliasForRemoval(virDomainObjPtr vm,
|
|
|
|
const char *alias)
|
2013-07-11 15:11:02 +00:00
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
|
2016-04-04 15:17:43 +00:00
|
|
|
memset(&priv->unplug, 0, sizeof(priv->unplug));
|
|
|
|
|
2016-08-04 21:53:13 +00:00
|
|
|
priv->unplug.alias = alias;
|
2013-07-11 15:11:02 +00:00
|
|
|
}
|
|
|
|
|
2016-08-04 21:53:13 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
qemuDomainMarkDeviceForRemoval(virDomainObjPtr vm,
|
|
|
|
virDomainDeviceInfoPtr info)
|
|
|
|
|
|
|
|
{
|
|
|
|
qemuDomainMarkDeviceAliasForRemoval(vm, info->alias);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-11 15:11:02 +00:00
|
|
|
static void
|
|
|
|
qemuDomainResetDeviceRemoval(virDomainObjPtr vm)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
2016-04-04 15:17:43 +00:00
|
|
|
priv->unplug.alias = NULL;
|
qemu_hotplug: Fix a rare race condition when detaching a device twice
https://bugzilla.redhat.com/show_bug.cgi?id=1623389
If a device is detached twice from the same domain the following
race condition may happen:
1) The first DetachDevice() call will issue "device_del" on qemu
monitor, but since the DEVICE_DELETED event did not arrive in
time, the API ends claiming "Device detach request sent
successfully".
2) The second DetachDevice() therefore still find the device in
the domain and thus proceeds to detaching it again. It calls
EnterMonitor() and qemuMonitorSend() trying to issue "device_del"
command again. This gets both domain lock and monitor lock
released.
3) At this point, qemu sends us the DEVICE_DELETED event which is
going to be handled by the event loop which ends up calling
qemuDomainSignalDeviceRemoval() to determine who is going to
remove the device from domain definition. Whether it is the
caller that marked the device for removal or whether it is going
to be the event processing thread.
4) Because the device was marked for removal,
qemuDomainSignalDeviceRemoval() returns true, which means the
event is to be processed by the thread that has marked the device
for removal (and is currently still trying to issue "device_del"
command)
5) The thread finally issues the "device_del" command, which
fails (obviously) and therefore it calls
qemuDomainResetDeviceRemoval() to reset the device marking and
quits immediately after, NOT removing any device from the domain
definition.
At this point, the device is still present in the domain
definition but doesn't exist in qemu anymore. Worse, there is no
way to remove it from the domain definition.
Solution is to note down that we've seen the event and if the
second "device_del" fails, not take it as a failure but carry on
with the usual execution.
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
ACKed-by: Peter Krempa <pkrempa@redhat.com>
2019-03-14 10:02:52 +00:00
|
|
|
priv->unplug.eventSeen = false;
|
2013-07-11 15:11:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns:
|
2016-04-04 15:17:43 +00:00
|
|
|
* -1 Unplug of the device failed
|
|
|
|
*
|
2019-02-07 11:08:57 +00:00
|
|
|
* 0 removal of the device did not finish in qemuDomainRemoveDeviceWaitTime
|
2016-04-04 13:08:40 +00:00
|
|
|
*
|
|
|
|
* 1 when the caller is responsible for finishing the device removal:
|
|
|
|
* - DEVICE_DELETED event arrived before the timeout time
|
|
|
|
* - we failed to reliably wait for the event and thus use fallback behavior
|
2013-07-11 15:11:02 +00:00
|
|
|
*/
|
|
|
|
static int
|
|
|
|
qemuDomainWaitForDeviceRemoval(virDomainObjPtr vm)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
unsigned long long until;
|
2016-04-04 11:59:48 +00:00
|
|
|
int rc;
|
2013-07-11 15:11:02 +00:00
|
|
|
|
|
|
|
if (virTimeMillisNow(&until) < 0)
|
2016-04-04 13:08:40 +00:00
|
|
|
return 1;
|
2013-07-26 10:18:01 +00:00
|
|
|
until += qemuDomainRemoveDeviceWaitTime;
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2016-04-04 15:17:43 +00:00
|
|
|
while (priv->unplug.alias) {
|
2016-04-04 11:59:48 +00:00
|
|
|
if ((rc = virDomainObjWaitUntil(vm, until)) == 1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (rc < 0) {
|
|
|
|
VIR_WARN("Failed to wait on unplug condition for domain '%s' "
|
2016-04-04 15:17:43 +00:00
|
|
|
"device '%s'", vm->def->name, priv->unplug.alias);
|
2016-04-04 11:59:48 +00:00
|
|
|
return 1;
|
2013-07-11 15:11:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-04 15:17:43 +00:00
|
|
|
if (priv->unplug.status == QEMU_DOMAIN_UNPLUGGING_DEVICE_STATUS_GUEST_REJECTED) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("unplug of device was rejected by the guest"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-07-11 15:11:02 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-05-26 15:01:52 +00:00
|
|
|
/* Returns:
|
|
|
|
* true there was a thread waiting for devAlias to be removed and this
|
|
|
|
* thread will take care of finishing the removal
|
|
|
|
* false the thread that started the removal is already gone and delegate
|
|
|
|
* finishing the removal to a new thread
|
|
|
|
*/
|
|
|
|
bool
|
2013-07-11 15:11:02 +00:00
|
|
|
qemuDomainSignalDeviceRemoval(virDomainObjPtr vm,
|
2016-04-04 15:17:43 +00:00
|
|
|
const char *devAlias,
|
|
|
|
qemuDomainUnpluggingDeviceStatus status)
|
2013-07-11 15:11:02 +00:00
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
|
2016-04-04 15:17:43 +00:00
|
|
|
if (STREQ_NULLABLE(priv->unplug.alias, devAlias)) {
|
2017-03-03 15:02:01 +00:00
|
|
|
VIR_DEBUG("Removal of device '%s' continues in waiting thread", devAlias);
|
2013-07-11 15:11:02 +00:00
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2016-04-04 15:17:43 +00:00
|
|
|
priv->unplug.status = status;
|
qemu_hotplug: Fix a rare race condition when detaching a device twice
https://bugzilla.redhat.com/show_bug.cgi?id=1623389
If a device is detached twice from the same domain the following
race condition may happen:
1) The first DetachDevice() call will issue "device_del" on qemu
monitor, but since the DEVICE_DELETED event did not arrive in
time, the API ends claiming "Device detach request sent
successfully".
2) The second DetachDevice() therefore still find the device in
the domain and thus proceeds to detaching it again. It calls
EnterMonitor() and qemuMonitorSend() trying to issue "device_del"
command again. This gets both domain lock and monitor lock
released.
3) At this point, qemu sends us the DEVICE_DELETED event which is
going to be handled by the event loop which ends up calling
qemuDomainSignalDeviceRemoval() to determine who is going to
remove the device from domain definition. Whether it is the
caller that marked the device for removal or whether it is going
to be the event processing thread.
4) Because the device was marked for removal,
qemuDomainSignalDeviceRemoval() returns true, which means the
event is to be processed by the thread that has marked the device
for removal (and is currently still trying to issue "device_del"
command)
5) The thread finally issues the "device_del" command, which
fails (obviously) and therefore it calls
qemuDomainResetDeviceRemoval() to reset the device marking and
quits immediately after, NOT removing any device from the domain
definition.
At this point, the device is still present in the domain
definition but doesn't exist in qemu anymore. Worse, there is no
way to remove it from the domain definition.
Solution is to note down that we've seen the event and if the
second "device_del" fails, not take it as a failure but carry on
with the usual execution.
Signed-off-by: Michal Privoznik <mprivozn@redhat.com>
ACKed-by: Peter Krempa <pkrempa@redhat.com>
2019-03-14 10:02:52 +00:00
|
|
|
priv->unplug.eventSeen = true;
|
2016-04-04 11:59:48 +00:00
|
|
|
virDomainObjBroadcast(vm);
|
2014-05-26 15:01:52 +00:00
|
|
|
return true;
|
2013-07-11 15:11:02 +00:00
|
|
|
}
|
2014-05-26 15:01:52 +00:00
|
|
|
return false;
|
2013-07-11 15:11:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-18 09:00:13 +00:00
|
|
|
static int
|
|
|
|
qemuDomainDetachVirtioDiskDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainDiskDefPtr detach,
|
|
|
|
bool async)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
int ret = -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2011-05-09 06:59:16 +00:00
|
|
|
if (qemuIsMultiFunctionDevice(vm->def, &detach->info)) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("cannot hot unplug multifunction PCI device: %s"),
|
2013-07-09 21:43:22 +00:00
|
|
|
detach->dst);
|
2011-05-09 06:59:16 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &detach->info);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, detach->info.alias) < 0) {
|
|
|
|
if (virDomainObjIsActive(vm))
|
|
|
|
virDomainAuditDisk(vm, detach->src, NULL, "detach", false);
|
2016-04-26 12:53:46 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveDiskDevice(driver, vm, detach);
|
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2010-12-16 16:10:54 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-07-18 09:00:13 +00:00
|
|
|
static int
|
|
|
|
qemuDomainDetachDiskDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainDiskDefPtr detach,
|
|
|
|
bool async)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
int ret = -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2016-06-09 11:16:10 +00:00
|
|
|
if (qemuDomainDiskBlockJobIsActive(detach))
|
blockjob: react to active block copy
For now, disk migration via block copy job is not implemented in
libvirt. But when we do implement it, we have to deal with the
fact that qemu does not yet provide an easy way to re-start a qemu
process with mirroring still intact. Paolo has proposed an idea
for a persistent dirty bitmap that might make this possible, but
until that design is complete, it's hard to say what changes
libvirt would need. Even something like 'virDomainSave' becomes
hairy, if you realize the implications that 'virDomainRestore'
would be stuck with recreating the same mirror layout.
But if we step back and look at the bigger picture, we realize that
the initial client of live storage migration via disk mirroring is
oVirt, which always uses transient domains, and that if a transient
domain is destroyed while a mirror exists, oVirt can easily restart
the storage migration by creating a new domain that visits just the
source storage, with no loss in data.
We can make life a lot easier by being cowards for now, forbidding
certain operations on a domain. This patch guarantees that we
never get in a state where we would have to restart a domain with
a mirroring block copy, by preventing saves, snapshots, migration,
hot unplug of a disk in use, and conversion to a persistent domain
(thankfully, it is still relatively easy to 'virsh undefine' a
running domain to temporarily make it transient, run tests on
'virsh blockcopy', then 'virsh define' to restore the persistence).
Later, if the qemu design is enhanced, we can relax our code.
The change to qemudDomainDefine looks a bit odd for undoing an
assignment, rather than probing up front to avoid the assignment,
but this is because of how virDomainAssignDef combines both a
lookup and assignment into a single function call.
* src/conf/domain_conf.h (virDomainHasDiskMirror): New prototype.
* src/conf/domain_conf.c (virDomainHasDiskMirror): New function.
* src/libvirt_private.syms (domain_conf.h): Export it.
* src/qemu/qemu_driver.c (qemuDomainSaveInternal)
(qemuDomainSnapshotCreateXML, qemuDomainRevertToSnapshot)
(qemuDomainBlockJobImpl, qemudDomainDefine): Prevent dangerous
actions while block copy is already in action.
* src/qemu/qemu_hotplug.c (qemuDomainDetachDiskDevice): Likewise.
* src/qemu/qemu_migration.c (qemuMigrationIsAllowed): Likewise.
2012-04-10 02:39:47 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &detach->info);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, detach->info.alias) < 0) {
|
|
|
|
if (virDomainObjIsActive(vm))
|
|
|
|
virDomainAuditDisk(vm, detach->src, NULL, "detach", false);
|
2010-12-16 16:10:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveDiskDevice(driver, vm, detach);
|
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2010-12-16 16:10:54 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-07-18 09:00:13 +00:00
|
|
|
static int
|
|
|
|
qemuFindDisk(virDomainDefPtr def, const char *dst)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
2014-11-13 14:25:30 +00:00
|
|
|
if (STREQ(def->disks[i]->dst, dst))
|
2013-07-18 09:00:13 +00:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainDetachDeviceDiskLive(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainDeviceDefPtr dev,
|
|
|
|
bool async)
|
2013-07-18 09:00:13 +00:00
|
|
|
{
|
|
|
|
virDomainDiskDefPtr disk;
|
|
|
|
int ret = -1;
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
if ((idx = qemuFindDisk(vm->def, dev->data.disk->dst)) < 0) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("disk %s not found"), dev->data.disk->dst);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
disk = vm->def->disks[idx];
|
|
|
|
|
|
|
|
switch (disk->device) {
|
|
|
|
case VIR_DOMAIN_DISK_DEVICE_DISK:
|
|
|
|
case VIR_DOMAIN_DISK_DEVICE_LUN:
|
|
|
|
if (disk->bus == VIR_DOMAIN_DISK_BUS_VIRTIO)
|
2018-05-23 16:01:16 +00:00
|
|
|
ret = qemuDomainDetachVirtioDiskDevice(driver, vm, disk, async);
|
2013-07-18 09:00:13 +00:00
|
|
|
else if (disk->bus == VIR_DOMAIN_DISK_BUS_SCSI ||
|
|
|
|
disk->bus == VIR_DOMAIN_DISK_BUS_USB)
|
2018-05-23 16:01:16 +00:00
|
|
|
ret = qemuDomainDetachDiskDevice(driver, vm, disk, async);
|
2013-07-18 09:00:13 +00:00
|
|
|
else
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("This type of disk cannot be hot unplugged"));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("disk device type '%s' cannot be detached"),
|
|
|
|
virDomainDiskDeviceTypeToString(disk->device));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-21 07:35:52 +00:00
|
|
|
static bool qemuDomainDiskControllerIsBusy(virDomainObjPtr vm,
|
|
|
|
virDomainControllerDefPtr detach)
|
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
size_t i;
|
2011-02-21 07:35:52 +00:00
|
|
|
virDomainDiskDefPtr disk;
|
qemu: Prevent detaching SCSI controller used by hostdev
Consider the following XML snippets:
$ cat scsicontroller.xml
<controller type='scsi' model='virtio-scsi' index='0'/>
$ cat scsihostdev.xml
<hostdev mode='subsystem' type='scsi'>
<source>
<adapter name='scsi_host0'/>
<address bus='0' target='8' unit='1074151456'/>
</source>
</hostdev>
If we create a guest that includes the contents of scsihostdev.xml,
but forget the virtio-scsi controller described in scsicontroller.xml,
one is silently created for us. The same holds true when attaching
a hostdev before the matching virtio-scsi controller.
(See qemuDomainFindOrCreateSCSIDiskController for context.)
Detaching the hostdev, followed by the controller, works well and the
guest behaves appropriately.
If we detach the virtio-scsi controller device first, any associated
hostdevs are detached for us by the underlying virtio-scsi code (this
is fine, since the connection is broken). But all is not well, as the
guest is unable to receive new virtio-scsi devices (the attach commands
succeed, but devices never appear within the guest), nor even be
shutdown, after this point.
While this is not libvirt's problem, we can prevent falling into this
scenario by checking if a controller is being used by any hostdev
devices. The same is already done for disk elements today.
Applying this patch and then using the XML snippets from earlier:
$ virsh detach-device guest_01 scsicontroller.xml
error: Failed to detach device from scsicontroller.xml
error: operation failed: device cannot be detached: device is busy
$ virsh detach-device guest_01 scsihostdev.xml
Device detached successfully
$ virsh detach-device guest_01 scsicontroller.xml
Device detached successfully
Signed-off-by: Eric Farman <farman@linux.vnet.ibm.com>
Reviewed-by: Bjoern Walk <bwalk@linux.vnet.ibm.com>
Reviewed-by: Boris Fiuczynski <fiuczy@linux.vnet.ibm.com>
2016-11-29 21:40:16 +00:00
|
|
|
virDomainHostdevDefPtr hostdev;
|
2011-02-21 07:35:52 +00:00
|
|
|
|
|
|
|
for (i = 0; i < vm->def->ndisks; i++) {
|
|
|
|
disk = vm->def->disks[i];
|
|
|
|
if (disk->info.type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_DRIVE)
|
|
|
|
/* the disk does not use disk controller */
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* check whether the disk uses this type controller */
|
|
|
|
if (disk->bus == VIR_DOMAIN_DISK_BUS_IDE &&
|
|
|
|
detach->type != VIR_DOMAIN_CONTROLLER_TYPE_IDE)
|
|
|
|
continue;
|
|
|
|
if (disk->bus == VIR_DOMAIN_DISK_BUS_FDC &&
|
|
|
|
detach->type != VIR_DOMAIN_CONTROLLER_TYPE_FDC)
|
|
|
|
continue;
|
|
|
|
if (disk->bus == VIR_DOMAIN_DISK_BUS_SCSI &&
|
|
|
|
detach->type != VIR_DOMAIN_CONTROLLER_TYPE_SCSI)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (disk->info.addr.drive.controller == detach->idx)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
qemu: Prevent detaching SCSI controller used by hostdev
Consider the following XML snippets:
$ cat scsicontroller.xml
<controller type='scsi' model='virtio-scsi' index='0'/>
$ cat scsihostdev.xml
<hostdev mode='subsystem' type='scsi'>
<source>
<adapter name='scsi_host0'/>
<address bus='0' target='8' unit='1074151456'/>
</source>
</hostdev>
If we create a guest that includes the contents of scsihostdev.xml,
but forget the virtio-scsi controller described in scsicontroller.xml,
one is silently created for us. The same holds true when attaching
a hostdev before the matching virtio-scsi controller.
(See qemuDomainFindOrCreateSCSIDiskController for context.)
Detaching the hostdev, followed by the controller, works well and the
guest behaves appropriately.
If we detach the virtio-scsi controller device first, any associated
hostdevs are detached for us by the underlying virtio-scsi code (this
is fine, since the connection is broken). But all is not well, as the
guest is unable to receive new virtio-scsi devices (the attach commands
succeed, but devices never appear within the guest), nor even be
shutdown, after this point.
While this is not libvirt's problem, we can prevent falling into this
scenario by checking if a controller is being used by any hostdev
devices. The same is already done for disk elements today.
Applying this patch and then using the XML snippets from earlier:
$ virsh detach-device guest_01 scsicontroller.xml
error: Failed to detach device from scsicontroller.xml
error: operation failed: device cannot be detached: device is busy
$ virsh detach-device guest_01 scsihostdev.xml
Device detached successfully
$ virsh detach-device guest_01 scsicontroller.xml
Device detached successfully
Signed-off-by: Eric Farman <farman@linux.vnet.ibm.com>
Reviewed-by: Bjoern Walk <bwalk@linux.vnet.ibm.com>
Reviewed-by: Boris Fiuczynski <fiuczy@linux.vnet.ibm.com>
2016-11-29 21:40:16 +00:00
|
|
|
for (i = 0; i < vm->def->nhostdevs; i++) {
|
|
|
|
hostdev = vm->def->hostdevs[i];
|
|
|
|
if (!virHostdevIsSCSIDevice(hostdev) ||
|
|
|
|
detach->type != VIR_DOMAIN_CONTROLLER_TYPE_SCSI)
|
|
|
|
continue;
|
|
|
|
if (hostdev->info->addr.drive.controller == detach->idx)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-02-21 07:35:52 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool qemuDomainControllerIsBusy(virDomainObjPtr vm,
|
|
|
|
virDomainControllerDefPtr detach)
|
|
|
|
{
|
|
|
|
switch (detach->type) {
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_IDE:
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_FDC:
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_SCSI:
|
|
|
|
return qemuDomainDiskControllerIsBusy(vm, detach);
|
|
|
|
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_SATA:
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_VIRTIO_SERIAL:
|
|
|
|
case VIR_DOMAIN_CONTROLLER_TYPE_CCID:
|
|
|
|
default:
|
|
|
|
/* libvirt does not support sata controller, and does not support to
|
|
|
|
* detach virtio and smart card controller.
|
|
|
|
*/
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-21 03:36:25 +00:00
|
|
|
int qemuDomainDetachControllerDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainDeviceDefPtr dev,
|
|
|
|
bool async)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
2012-07-23 08:18:57 +00:00
|
|
|
int idx, ret = -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
virDomainControllerDefPtr detach = NULL;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
|
2012-07-23 08:18:57 +00:00
|
|
|
if ((idx = virDomainControllerFind(vm->def,
|
|
|
|
dev->data.controller->type,
|
|
|
|
dev->data.controller->idx)) < 0) {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2013-05-09 09:39:21 +00:00
|
|
|
_("controller %s:%d not found"),
|
2012-07-18 15:22:03 +00:00
|
|
|
virDomainControllerTypeToString(dev->data.controller->type),
|
|
|
|
dev->data.controller->idx);
|
2010-12-16 16:10:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2012-07-23 08:18:57 +00:00
|
|
|
detach = vm->def->controllers[idx];
|
|
|
|
|
2013-11-21 03:36:26 +00:00
|
|
|
if (detach->info.type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI &&
|
|
|
|
detach->info.type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_CCW &&
|
|
|
|
detach->info.type != VIR_DOMAIN_DEVICE_ADDRESS_TYPE_VIRTIO_S390) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("device with '%s' address cannot be detached"),
|
|
|
|
virDomainDeviceAddressTypeToString(detach->info.type));
|
2010-12-16 16:10:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2013-11-21 03:36:26 +00:00
|
|
|
if (!virDomainDeviceAddressIsValid(&detach->info, detach->info.type)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("device with invalid '%s' address cannot be detached"),
|
|
|
|
virDomainDeviceAddressTypeToString(detach->info.type));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2017-10-18 13:33:45 +00:00
|
|
|
if (qemuIsMultiFunctionDevice(vm->def, &detach->info)) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("cannot hot unplug multifunction PCI device: %s"),
|
|
|
|
dev->data.disk->dst);
|
2011-05-09 06:59:16 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2011-02-21 07:35:52 +00:00
|
|
|
if (qemuDomainControllerIsBusy(vm, detach)) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("device cannot be detached: device is busy"));
|
2011-02-21 07:35:52 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &detach->info);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (detach->type == VIR_DOMAIN_CONTROLLER_TYPE_PCI) {
|
|
|
|
int rc;
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuDomainDetachExtensionDevice(priv->mon, &detach->info);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
rc = -1;
|
2018-11-08 11:00:30 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (rc < 0)
|
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
2018-11-08 11:00:30 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, detach->info.alias) < 0)
|
2014-12-16 14:50:20 +00:00
|
|
|
goto cleanup;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveControllerDevice(driver, vm, detach);
|
|
|
|
}
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2010-12-16 16:10:54 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-12-15 16:51:56 +00:00
|
|
|
static int
|
2019-03-12 12:49:24 +00:00
|
|
|
qemuDomainDetachHostPCIDevice(virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainHostdevDefPtr detach,
|
|
|
|
bool async)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
2014-07-03 20:31:39 +00:00
|
|
|
virDomainHostdevSubsysPCIPtr pcisrc = &detach->source.subsys.u.pci;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2012-01-31 09:16:54 +00:00
|
|
|
if (qemuIsMultiFunctionDevice(vm->def, detach->info)) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
_("cannot hot unplug multifunction PCI device: %.4x:%.2x:%.2x.%.1x"),
|
2014-07-03 20:31:39 +00:00
|
|
|
pcisrc->addr.domain, pcisrc->addr.bus,
|
|
|
|
pcisrc->addr.slot, pcisrc->addr.function);
|
2013-07-11 09:14:16 +00:00
|
|
|
return -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, detach->info);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
return qemuDomainDeleteDevice(vm, detach->info->alias);
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2011-12-15 16:51:56 +00:00
|
|
|
static int
|
2019-03-12 12:49:24 +00:00
|
|
|
qemuDomainDetachHostUSBDevice(virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainHostdevDefPtr detach,
|
|
|
|
bool async)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
2012-01-31 09:16:54 +00:00
|
|
|
if (!detach->info->alias) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
"%s", _("device cannot be detached without a device alias"));
|
2010-12-16 16:10:54 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, detach->info);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
return qemuDomainDeleteDevice(vm, detach->info->alias);
|
2010-12-16 16:10:54 +00:00
|
|
|
}
|
|
|
|
|
2013-05-03 18:07:31 +00:00
|
|
|
static int
|
2019-03-12 12:49:24 +00:00
|
|
|
qemuDomainDetachHostSCSIDevice(virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainHostdevDefPtr detach,
|
|
|
|
bool async)
|
2013-05-03 18:07:31 +00:00
|
|
|
{
|
|
|
|
if (!detach->info->alias) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
"%s", _("device cannot be detached without a device alias"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, detach->info);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
return qemuDomainDeleteDevice(vm, detach->info->alias);
|
2013-05-03 18:07:31 +00:00
|
|
|
}
|
|
|
|
|
2016-11-22 03:58:19 +00:00
|
|
|
static int
|
2019-03-12 12:49:24 +00:00
|
|
|
qemuDomainDetachSCSIVHostDevice(virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainHostdevDefPtr detach,
|
|
|
|
bool async)
|
2016-11-22 03:58:19 +00:00
|
|
|
{
|
|
|
|
if (!detach->info->alias) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
|
|
|
"%s", _("device cannot be detached without a device alias"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, detach->info);
|
2016-11-22 03:58:19 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
return qemuDomainDeleteDevice(vm, detach->info->alias);
|
2016-11-22 03:58:19 +00:00
|
|
|
}
|
|
|
|
|
2018-03-26 08:06:07 +00:00
|
|
|
|
|
|
|
static int
|
2019-03-12 12:49:24 +00:00
|
|
|
qemuDomainDetachMediatedDevice(virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainHostdevDefPtr detach,
|
|
|
|
bool async)
|
2018-03-26 08:06:07 +00:00
|
|
|
{
|
|
|
|
if (!detach->info->alias) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("device cannot be detached without a device alias"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, detach->info);
|
2018-03-26 08:06:07 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
return qemuDomainDeleteDevice(vm, detach->info->alias);
|
2018-03-26 08:06:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-03 18:07:31 +00:00
|
|
|
static int
|
2014-09-24 12:31:32 +00:00
|
|
|
qemuDomainDetachThisHostDevice(virQEMUDriverPtr driver,
|
2013-05-03 18:07:31 +00:00
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainHostdevDefPtr detach,
|
|
|
|
bool async)
|
2010-12-16 16:10:54 +00:00
|
|
|
{
|
2012-02-27 11:53:19 +00:00
|
|
|
int ret = -1;
|
2010-12-16 16:10:54 +00:00
|
|
|
|
2017-10-14 17:16:20 +00:00
|
|
|
if (qemuAssignDeviceHostdevAlias(vm->def, &detach->info->alias, -1) < 0)
|
|
|
|
return -1;
|
2014-10-08 22:45:39 +00:00
|
|
|
|
2012-02-27 11:53:19 +00:00
|
|
|
switch (detach->source.subsys.type) {
|
2010-12-16 16:10:54 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI:
|
2019-03-12 12:49:24 +00:00
|
|
|
ret = qemuDomainDetachHostPCIDevice(vm, detach, async);
|
2012-03-07 01:43:22 +00:00
|
|
|
break;
|
2010-12-16 16:10:54 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB:
|
2019-03-12 12:49:24 +00:00
|
|
|
ret = qemuDomainDetachHostUSBDevice(vm, detach, async);
|
2010-12-16 16:10:54 +00:00
|
|
|
break;
|
2013-05-03 18:07:31 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI:
|
2019-03-12 12:49:24 +00:00
|
|
|
ret = qemuDomainDetachHostSCSIDevice(vm, detach, async);
|
2013-05-03 18:07:31 +00:00
|
|
|
break;
|
2016-11-22 03:58:19 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI_HOST:
|
2019-03-12 12:49:24 +00:00
|
|
|
ret = qemuDomainDetachSCSIVHostDevice(vm, detach, async);
|
2016-11-22 03:58:19 +00:00
|
|
|
break;
|
2018-03-26 08:06:07 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_MDEV:
|
2019-03-12 12:49:24 +00:00
|
|
|
ret = qemuDomainDetachMediatedDevice(vm, detach, async);
|
2018-03-26 08:06:07 +00:00
|
|
|
break;
|
2010-12-16 16:10:54 +00:00
|
|
|
default:
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
2017-05-11 13:26:47 +00:00
|
|
|
_("hot unplug is not supported for hostdev subsys type '%s'"),
|
2012-07-18 15:22:03 +00:00
|
|
|
virDomainHostdevSubsysTypeToString(detach->source.subsys.type));
|
2010-12-16 16:10:54 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-05-26 15:01:52 +00:00
|
|
|
if (ret < 0) {
|
2014-12-16 14:50:20 +00:00
|
|
|
if (virDomainObjIsActive(vm))
|
|
|
|
virDomainAuditHostdev(vm, detach, "detach", false);
|
2018-05-23 16:01:16 +00:00
|
|
|
} else if (!async &&
|
|
|
|
(ret = qemuDomainWaitForDeviceRemoval(vm)) == 1) {
|
2016-04-04 13:08:40 +00:00
|
|
|
ret = qemuDomainRemoveHostDevice(driver, vm, detach);
|
2014-05-26 15:01:52 +00:00
|
|
|
}
|
2013-04-29 17:15:26 +00:00
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2010-12-16 16:10:54 +00:00
|
|
|
return ret;
|
|
|
|
}
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
|
2012-02-27 11:53:19 +00:00
|
|
|
/* search for a hostdev matching dev and detach it */
|
2014-09-24 12:31:32 +00:00
|
|
|
int qemuDomainDetachHostDevice(virQEMUDriverPtr driver,
|
2012-02-27 11:53:19 +00:00
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainDeviceDefPtr dev,
|
|
|
|
bool async)
|
2012-02-27 11:53:19 +00:00
|
|
|
{
|
|
|
|
virDomainHostdevDefPtr hostdev = dev->data.hostdev;
|
|
|
|
virDomainHostdevSubsysPtr subsys = &hostdev->source.subsys;
|
2014-07-03 19:43:05 +00:00
|
|
|
virDomainHostdevSubsysUSBPtr usbsrc = &subsys->u.usb;
|
2014-07-03 20:31:39 +00:00
|
|
|
virDomainHostdevSubsysPCIPtr pcisrc = &subsys->u.pci;
|
2014-07-03 21:01:10 +00:00
|
|
|
virDomainHostdevSubsysSCSIPtr scsisrc = &subsys->u.scsi;
|
2018-03-26 08:06:07 +00:00
|
|
|
virDomainHostdevSubsysMediatedDevPtr mdevsrc = &subsys->u.mdev;
|
2012-02-27 11:53:19 +00:00
|
|
|
virDomainHostdevDefPtr detach = NULL;
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
if (hostdev->mode != VIR_DOMAIN_HOSTDEV_MODE_SUBSYS) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
2017-05-11 13:26:47 +00:00
|
|
|
_("hot unplug is not supported for hostdev mode '%s'"),
|
2012-07-18 15:22:03 +00:00
|
|
|
virDomainHostdevModeTypeToString(hostdev->mode));
|
2012-02-27 11:53:19 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
idx = virDomainHostdevFind(vm->def, hostdev, &detach);
|
|
|
|
|
|
|
|
if (idx < 0) {
|
2012-10-17 09:23:12 +00:00
|
|
|
switch (subsys->type) {
|
2012-02-27 11:53:19 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_PCI:
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2012-07-18 15:22:03 +00:00
|
|
|
_("host pci device %.4x:%.2x:%.2x.%.1x not found"),
|
2014-07-03 20:31:39 +00:00
|
|
|
pcisrc->addr.domain, pcisrc->addr.bus,
|
|
|
|
pcisrc->addr.slot, pcisrc->addr.function);
|
2012-02-27 11:53:19 +00:00
|
|
|
break;
|
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB:
|
2014-07-03 19:43:05 +00:00
|
|
|
if (usbsrc->bus && usbsrc->device) {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2012-07-18 15:22:03 +00:00
|
|
|
_("host usb device %03d.%03d not found"),
|
2014-07-03 19:43:05 +00:00
|
|
|
usbsrc->bus, usbsrc->device);
|
2012-02-27 11:53:19 +00:00
|
|
|
} else {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2012-07-18 15:22:03 +00:00
|
|
|
_("host usb device vendor=0x%.4x product=0x%.4x not found"),
|
2014-07-03 19:43:05 +00:00
|
|
|
usbsrc->vendor, usbsrc->product);
|
2012-02-27 11:53:19 +00:00
|
|
|
}
|
|
|
|
break;
|
2014-06-20 15:35:46 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI: {
|
2014-07-09 13:31:38 +00:00
|
|
|
if (scsisrc->protocol ==
|
|
|
|
VIR_DOMAIN_HOSTDEV_SCSI_PROTOCOL_TYPE_ISCSI) {
|
|
|
|
virDomainHostdevSubsysSCSIiSCSIPtr iscsisrc = &scsisrc->u.iscsi;
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2014-07-09 13:31:38 +00:00
|
|
|
_("host scsi iSCSI path %s not found"),
|
2017-09-22 19:18:22 +00:00
|
|
|
iscsisrc->src->path);
|
2014-07-09 13:31:38 +00:00
|
|
|
} else {
|
|
|
|
virDomainHostdevSubsysSCSIHostPtr scsihostsrc =
|
|
|
|
&scsisrc->u.host;
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2015-06-17 03:29:53 +00:00
|
|
|
_("host scsi device %s:%u:%u.%llu not found"),
|
2014-07-09 13:31:38 +00:00
|
|
|
scsihostsrc->adapter, scsihostsrc->bus,
|
|
|
|
scsihostsrc->target, scsihostsrc->unit);
|
|
|
|
}
|
2013-05-03 18:07:31 +00:00
|
|
|
break;
|
2014-06-20 15:35:46 +00:00
|
|
|
}
|
2018-03-26 08:06:07 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_MDEV:
|
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
|
|
|
_("mediated device '%s' not found"),
|
|
|
|
mdevsrc->uuidstr);
|
|
|
|
break;
|
2016-11-22 03:58:19 +00:00
|
|
|
case VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_SCSI_HOST:
|
|
|
|
break;
|
2012-02-27 11:53:19 +00:00
|
|
|
default:
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("unexpected hostdev type %d"), subsys->type);
|
2012-02-27 11:53:19 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
qemu: support type=hostdev network device live hotplug attach/detach
qemuDomainAttachNetDevice
- re-ordered some things at start of function because
networkAllocateActualDevice should always be run and a slot
in def->nets always allocated, but host_net_add isn't needed
if the actual type is hostdev.
- if actual type is hostdev, defer to
qemuDomainAttachHostDevice (which will reach up to the NetDef
for things like MAC address when necessary). After return
from qemuDomainAttachHostDevice, slip directly to cleanup,
since the rest of the function is specific to emulated net
devices.
- put assignment of new NetDef into expanded def->nets down
below cleanup: (but only on success) since it is also needed
for emulated and hostdev net devices.
qemuDomainDetachHostDevice
- after locating the exact device to detach, check if it's a
network device and, if so, use toplevel
qemuDomainDetachNetDevice instead so that the def->nets list
is properly updated, and 'actual device' properly returned to
network pool if appropriate. Otherwise, for normal hostdevs,
call the lower level qemuDomainDetachThisDevice.
qemuDomainDetachNetDevice
- This is where it gets a bit tricky. After locating the device
on the def->nets list, if the network device type == hostdev,
call the *lower level* qemuDomainDetachThisDevice (which will
reach back up to the parent net device for MAC address /
virtualport when appropriate, then clear the device out of
def->hostdevs) before skipping past all the emulated
net-device-specific code to cleanup:, where the network
device is removed from def->nets, and the network device
object is freed.
In short, any time a hostdev-type network device is detached, we must
go through the toplevel virDomaineDetachNetDevice function first and
last, to make sure 1) the def->nnets list is properly managed, and 2)
any device allocated with networkAllocateActualDevice is properly
freed. At the same time, in the middle we need to go through the
lower-level vidDomainDetach*This*HostDevice to be sure that 1) the
def->hostdevs list is properly managed, 2) the PCI device is properly
detached from the guest and reattached to the host (if appropriate),
and 3) any higher level teardown is called at the appropriate time, by
reaching back up to the NetDef config (part (3) will be covered in a
separate patch).
2012-02-27 19:20:17 +00:00
|
|
|
/* If this is a network hostdev, we need to use the higher-level detach
|
|
|
|
* function so that mac address / virtualport are reset
|
|
|
|
*/
|
|
|
|
if (detach->parent.type == VIR_DOMAIN_DEVICE_NET)
|
2018-05-23 16:01:16 +00:00
|
|
|
return qemuDomainDetachNetDevice(driver, vm, &detach->parent, async);
|
qemu: support type=hostdev network device live hotplug attach/detach
qemuDomainAttachNetDevice
- re-ordered some things at start of function because
networkAllocateActualDevice should always be run and a slot
in def->nets always allocated, but host_net_add isn't needed
if the actual type is hostdev.
- if actual type is hostdev, defer to
qemuDomainAttachHostDevice (which will reach up to the NetDef
for things like MAC address when necessary). After return
from qemuDomainAttachHostDevice, slip directly to cleanup,
since the rest of the function is specific to emulated net
devices.
- put assignment of new NetDef into expanded def->nets down
below cleanup: (but only on success) since it is also needed
for emulated and hostdev net devices.
qemuDomainDetachHostDevice
- after locating the exact device to detach, check if it's a
network device and, if so, use toplevel
qemuDomainDetachNetDevice instead so that the def->nets list
is properly updated, and 'actual device' properly returned to
network pool if appropriate. Otherwise, for normal hostdevs,
call the lower level qemuDomainDetachThisDevice.
qemuDomainDetachNetDevice
- This is where it gets a bit tricky. After locating the device
on the def->nets list, if the network device type == hostdev,
call the *lower level* qemuDomainDetachThisDevice (which will
reach back up to the parent net device for MAC address /
virtualport when appropriate, then clear the device out of
def->hostdevs) before skipping past all the emulated
net-device-specific code to cleanup:, where the network
device is removed from def->nets, and the network device
object is freed.
In short, any time a hostdev-type network device is detached, we must
go through the toplevel virDomaineDetachNetDevice function first and
last, to make sure 1) the def->nnets list is properly managed, and 2)
any device allocated with networkAllocateActualDevice is properly
freed. At the same time, in the middle we need to go through the
lower-level vidDomainDetach*This*HostDevice to be sure that 1) the
def->hostdevs list is properly managed, 2) the PCI device is properly
detached from the guest and reattached to the host (if appropriate),
and 3) any higher level teardown is called at the appropriate time, by
reaching back up to the NetDef config (part (3) will be covered in a
separate patch).
2012-02-27 19:20:17 +00:00
|
|
|
else
|
2018-05-23 16:01:16 +00:00
|
|
|
return qemuDomainDetachThisHostDevice(driver, vm, detach, async);
|
2012-02-27 11:53:19 +00:00
|
|
|
}
|
|
|
|
|
2016-09-12 13:40:48 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainDetachShmemDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainShmemDefPtr dev,
|
|
|
|
bool async)
|
2016-09-12 13:40:48 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
ssize_t idx = -1;
|
|
|
|
virDomainShmemDefPtr shmem = NULL;
|
|
|
|
|
|
|
|
if ((idx = virDomainShmemDefFind(vm->def, dev)) < 0) {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2018-01-23 04:24:42 +00:00
|
|
|
_("model '%s' shmem device not present "
|
|
|
|
"in domain configuration"),
|
|
|
|
virDomainShmemModelTypeToString(dev->model));
|
2016-09-12 13:40:48 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
shmem = vm->def->shmems[idx];
|
|
|
|
|
|
|
|
switch ((virDomainShmemModel)shmem->model) {
|
|
|
|
case VIR_DOMAIN_SHMEM_MODEL_IVSHMEM_PLAIN:
|
|
|
|
case VIR_DOMAIN_SHMEM_MODEL_IVSHMEM_DOORBELL:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_SHMEM_MODEL_IVSHMEM:
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("live detach of shmem model '%s' is not supported"),
|
|
|
|
virDomainShmemModelTypeToString(shmem->model));
|
2017-06-07 08:46:41 +00:00
|
|
|
ATTRIBUTE_FALLTHROUGH;
|
2016-09-12 13:40:48 +00:00
|
|
|
case VIR_DOMAIN_SHMEM_MODEL_LAST:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &shmem->info);
|
2016-09-12 13:40:48 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, shmem->info.alias) < 0)
|
2018-05-23 16:36:57 +00:00
|
|
|
goto cleanup;
|
2016-09-12 13:40:48 +00:00
|
|
|
|
2018-08-17 07:11:34 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveShmemDevice(driver, vm, shmem);
|
|
|
|
}
|
2016-09-12 13:40:48 +00:00
|
|
|
|
2018-05-23 16:36:57 +00:00
|
|
|
cleanup:
|
2018-08-17 07:11:34 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2016-09-12 13:40:48 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-05 09:08:36 +00:00
|
|
|
int
|
|
|
|
qemuDomainDetachWatchdog(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainWatchdogDefPtr dev,
|
|
|
|
bool async)
|
2017-09-05 09:08:36 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
virDomainWatchdogDefPtr watchdog = vm->def->watchdog;
|
|
|
|
|
2018-02-09 13:52:10 +00:00
|
|
|
if (!watchdog) {
|
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING, "%s",
|
|
|
|
_("watchdog device not present in domain configuration"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-09-05 09:08:36 +00:00
|
|
|
/* While domains can have up to one watchdog, the one supplied by the user
|
|
|
|
* doesn't necessarily match the one domain has. Refuse to detach in such
|
|
|
|
* case. */
|
2018-02-09 13:52:10 +00:00
|
|
|
if (!(watchdog->model == dev->model &&
|
2017-09-05 09:08:36 +00:00
|
|
|
watchdog->action == dev->action &&
|
|
|
|
virDomainDeviceInfoAddressIsEqual(&dev->info, &watchdog->info))) {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2018-01-23 04:24:42 +00:00
|
|
|
_("model '%s' watchdog device not present "
|
|
|
|
"in domain configuration"),
|
|
|
|
virDomainWatchdogModelTypeToString(watchdog->model));
|
2017-09-05 09:08:36 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (watchdog->model != VIR_DOMAIN_WATCHDOG_MODEL_I6300ESB) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("hot unplug of watchdog of model %s is not supported"),
|
|
|
|
virDomainWatchdogModelTypeToString(watchdog->model));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &watchdog->info);
|
2017-09-05 09:08:36 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, watchdog->info.alias) < 0)
|
2018-05-23 16:36:57 +00:00
|
|
|
goto cleanup;
|
2017-09-05 09:08:36 +00:00
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveWatchdog(driver, vm, watchdog);
|
|
|
|
}
|
2017-09-05 09:08:36 +00:00
|
|
|
|
2018-05-23 16:36:57 +00:00
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2017-09-05 09:08:36 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-01-05 02:47:47 +00:00
|
|
|
int
|
|
|
|
qemuDomainDetachRedirdevDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainRedirdevDefPtr dev,
|
|
|
|
bool async)
|
2018-01-05 02:47:47 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
virDomainRedirdevDefPtr tmpRedirdevDef;
|
|
|
|
ssize_t idx;
|
|
|
|
|
|
|
|
if ((idx = virDomainRedirdevDefFind(vm->def, dev)) < 0) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_INVALID, "%s",
|
|
|
|
_("no matching redirdev was not found"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmpRedirdevDef = vm->def->redirdevs[idx];
|
|
|
|
|
|
|
|
if (!tmpRedirdevDef->info.alias) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("alias not set for redirdev device"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &tmpRedirdevDef->info);
|
2018-01-05 02:47:47 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, tmpRedirdevDef->info.alias) < 0)
|
2018-01-05 02:47:47 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveRedirdevDevice(driver, vm, tmpRedirdevDef);
|
|
|
|
}
|
2018-01-05 02:47:47 +00:00
|
|
|
|
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2018-01-05 02:47:47 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-27 12:03:12 +00:00
|
|
|
int
|
2014-09-24 12:31:32 +00:00
|
|
|
qemuDomainDetachNetDevice(virQEMUDriverPtr driver,
|
2012-02-27 12:03:12 +00:00
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainDeviceDefPtr dev,
|
|
|
|
bool async)
|
2012-02-27 12:03:12 +00:00
|
|
|
{
|
qemu: fix attach/detach of netdevs with matching mac addrs
This resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=862515
which describes inconsistencies in dealing with duplicate mac
addresses on network devices in a domain.
(at any rate, it resolves *almost* everything, and prints out an
informative error message for the one problem that isn't solved, but
has a workaround.)
A synopsis of the problems:
1) you can't do a persistent attach-interface of a device with a mac
address that matches an existing device.
2) you *can* do a live attach-interface of such a device.
3) you *can* directly edit a domain and put in two devices with
matching mac addresses.
4) When running virsh detach-device (live or config), only MAC address
is checked when matching the device to remove, so the first device
with the desired mac address will be removed. This isn't always the
one that's wanted.
5) when running virsh detach-interface (live or config), the only two
items that can be specified to match against are mac address and model
type (virtio, etc) - if multiple netdevs match both of those
attributes, it again just finds the first one added and assumes that
is the only match.
Since it is completely valid to have multiple network devices with the
same MAC address (although it can cause problems in many cases, there
*are* valid use cases), what is needed is:
1) remove the restriction that prohibits doing a persistent add of a
netdev with a duplicate mac address.
2) enhance the backend of virDomainDetachDeviceFlags to check for
something that *is* guaranteed unique (but still work with just mac
address, as long as it yields only a single results.
This patch does three things:
1) removes the check for duplicate mac address during a persistent
netdev attach.
2) unifies the searching for both live and config detach of netdevices
in the subordinate functions of qemuDomainModifyDeviceFlags() to use the
new function virDomainNetFindIdx (which matches mac address and PCI
address if available, checking for duplicates if only mac address was
specified). This function returns -2 if multiple matches are found,
allowing the callers to print out an appropriate message.
Steps 1 & 2 are enough to fully fix the problem when using virsh
attach-device and detach-device (which require an XML description of
the device rather than a bunch of commandline args)
3) modifies the virsh detach-interface command to check for multiple
matches of mac address and show an error message suggesting use of the
detach-device command in cases where there are multiple matching mac
addresses.
Later we should decide how we want to input a PCI address on the virsh
commandline, and enhance detach-interface to take a --address option,
eliminating the need to use detach-device
* src/conf/domain_conf.c
* src/conf/domain_conf.h
* src/libvirt_private.syms
* added new virDomainNetFindIdx function
* removed now unused virDomainNetIndexByMac and
virDomainNetRemoveByMac
* src/qemu/qemu_driver.c
* remove check for duplicate max from qemuDomainAttachDeviceConfig
* use virDomainNetFindIdx/virDomainNetRemove instead
of virDomainNetRemoveByMac in qemuDomainDetachDeviceConfig
* use virDomainNetFindIdx instead of virDomainIndexByMac
in qemuDomainUpdateDeviceConfig
* src/qemu/qemu_hotplug.c
* use virDomainNetFindIdx instead of a homespun loop in
qemuDomainDetachNetDevice.
* tools/virsh-domain.c: modified detach-interface command as described
above
2012-10-25 20:03:35 +00:00
|
|
|
int detachidx, ret = -1;
|
2012-02-27 12:03:12 +00:00
|
|
|
virDomainNetDefPtr detach = NULL;
|
|
|
|
|
2014-04-01 13:56:55 +00:00
|
|
|
if ((detachidx = virDomainNetFindIdx(vm->def, dev->data.net)) < 0)
|
2012-02-27 12:03:12 +00:00
|
|
|
goto cleanup;
|
2014-04-01 13:56:55 +00:00
|
|
|
|
qemu: fix attach/detach of netdevs with matching mac addrs
This resolves:
https://bugzilla.redhat.com/show_bug.cgi?id=862515
which describes inconsistencies in dealing with duplicate mac
addresses on network devices in a domain.
(at any rate, it resolves *almost* everything, and prints out an
informative error message for the one problem that isn't solved, but
has a workaround.)
A synopsis of the problems:
1) you can't do a persistent attach-interface of a device with a mac
address that matches an existing device.
2) you *can* do a live attach-interface of such a device.
3) you *can* directly edit a domain and put in two devices with
matching mac addresses.
4) When running virsh detach-device (live or config), only MAC address
is checked when matching the device to remove, so the first device
with the desired mac address will be removed. This isn't always the
one that's wanted.
5) when running virsh detach-interface (live or config), the only two
items that can be specified to match against are mac address and model
type (virtio, etc) - if multiple netdevs match both of those
attributes, it again just finds the first one added and assumes that
is the only match.
Since it is completely valid to have multiple network devices with the
same MAC address (although it can cause problems in many cases, there
*are* valid use cases), what is needed is:
1) remove the restriction that prohibits doing a persistent add of a
netdev with a duplicate mac address.
2) enhance the backend of virDomainDetachDeviceFlags to check for
something that *is* guaranteed unique (but still work with just mac
address, as long as it yields only a single results.
This patch does three things:
1) removes the check for duplicate mac address during a persistent
netdev attach.
2) unifies the searching for both live and config detach of netdevices
in the subordinate functions of qemuDomainModifyDeviceFlags() to use the
new function virDomainNetFindIdx (which matches mac address and PCI
address if available, checking for duplicates if only mac address was
specified). This function returns -2 if multiple matches are found,
allowing the callers to print out an appropriate message.
Steps 1 & 2 are enough to fully fix the problem when using virsh
attach-device and detach-device (which require an XML description of
the device rather than a bunch of commandline args)
3) modifies the virsh detach-interface command to check for multiple
matches of mac address and show an error message suggesting use of the
detach-device command in cases where there are multiple matching mac
addresses.
Later we should decide how we want to input a PCI address on the virsh
commandline, and enhance detach-interface to take a --address option,
eliminating the need to use detach-device
* src/conf/domain_conf.c
* src/conf/domain_conf.h
* src/libvirt_private.syms
* added new virDomainNetFindIdx function
* removed now unused virDomainNetIndexByMac and
virDomainNetRemoveByMac
* src/qemu/qemu_driver.c
* remove check for duplicate max from qemuDomainAttachDeviceConfig
* use virDomainNetFindIdx/virDomainNetRemove instead
of virDomainNetRemoveByMac in qemuDomainDetachDeviceConfig
* use virDomainNetFindIdx instead of virDomainIndexByMac
in qemuDomainUpdateDeviceConfig
* src/qemu/qemu_hotplug.c
* use virDomainNetFindIdx instead of a homespun loop in
qemuDomainDetachNetDevice.
* tools/virsh-domain.c: modified detach-interface command as described
above
2012-10-25 20:03:35 +00:00
|
|
|
detach = vm->def->nets[detachidx];
|
2012-02-27 12:03:12 +00:00
|
|
|
|
2012-03-12 15:50:02 +00:00
|
|
|
if (virDomainNetGetActualType(detach) == VIR_DOMAIN_NET_TYPE_HOSTDEV) {
|
2014-09-24 12:31:32 +00:00
|
|
|
ret = qemuDomainDetachThisHostDevice(driver, vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainNetGetActualHostdev(detach),
|
|
|
|
async);
|
2012-03-12 15:50:02 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2017-10-18 13:39:33 +00:00
|
|
|
if (qemuIsMultiFunctionDevice(vm->def, &detach->info)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED,
|
2017-10-19 12:46:13 +00:00
|
|
|
_("cannot hot unplug multifunction PCI device: %s"),
|
2018-10-19 15:27:48 +00:00
|
|
|
detach->ifname);
|
2017-10-18 13:39:33 +00:00
|
|
|
goto cleanup;
|
2012-02-27 12:03:12 +00:00
|
|
|
}
|
|
|
|
|
2016-04-26 13:09:49 +00:00
|
|
|
if (!detach->info.alias) {
|
2014-10-08 22:55:43 +00:00
|
|
|
if (qemuAssignDeviceNetAlias(vm->def, detach, -1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2015-02-24 17:12:56 +00:00
|
|
|
if (virDomainNetGetActualBandwidth(detach) &&
|
|
|
|
virNetDevSupportBandwidth(virDomainNetGetActualType(detach)) &&
|
2014-11-18 23:55:48 +00:00
|
|
|
virNetDevBandwidthClear(detach->ifname) < 0)
|
|
|
|
VIR_WARN("cannot clear bandwidth setting for device : %s",
|
|
|
|
detach->ifname);
|
|
|
|
|
2015-04-13 17:26:54 +00:00
|
|
|
/* deactivate the tap/macvtap device on the host, which could also
|
|
|
|
* affect the parent device (e.g. macvtap passthrough mode sets
|
|
|
|
* the parent device offline)
|
2014-12-11 20:11:10 +00:00
|
|
|
*/
|
|
|
|
ignore_value(qemuInterfaceStopDevice(detach));
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &detach->info);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, detach->info.alias) < 0) {
|
|
|
|
if (virDomainObjIsActive(vm))
|
|
|
|
virDomainAuditNet(vm, detach, NULL, "detach", false);
|
2016-04-26 13:09:49 +00:00
|
|
|
goto cleanup;
|
2012-02-27 12:03:12 +00:00
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveNetDevice(driver, vm, detach);
|
|
|
|
}
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2012-02-27 12:03:12 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
int
|
2012-11-28 16:43:10 +00:00
|
|
|
qemuDomainChangeGraphicsPasswords(virQEMUDriverPtr driver,
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
int type,
|
|
|
|
virDomainGraphicsAuthDefPtr auth,
|
2014-08-12 02:54:42 +00:00
|
|
|
const char *defaultPasswd,
|
|
|
|
int asyncJob)
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
time_t now = time(NULL);
|
2016-06-29 15:14:52 +00:00
|
|
|
const char *expire;
|
|
|
|
char *validTo = NULL;
|
2011-05-26 14:15:54 +00:00
|
|
|
const char *connected = NULL;
|
2016-06-28 12:39:58 +00:00
|
|
|
const char *password;
|
2013-01-10 21:03:14 +00:00
|
|
|
int ret = -1;
|
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
|
2013-04-19 05:00:44 +00:00
|
|
|
if (!auth->passwd && !defaultPasswd) {
|
2013-01-10 21:03:14 +00:00
|
|
|
ret = 0;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2016-06-28 12:39:58 +00:00
|
|
|
password = auth->passwd ? auth->passwd : defaultPasswd;
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
|
2011-05-26 14:15:54 +00:00
|
|
|
if (auth->connected)
|
|
|
|
connected = virDomainGraphicsAuthConnectedTypeToString(auth->connected);
|
|
|
|
|
2014-08-12 02:54:42 +00:00
|
|
|
if (qemuDomainObjEnterMonitorAsync(driver, vm, asyncJob) < 0)
|
|
|
|
goto cleanup;
|
2016-06-28 12:39:58 +00:00
|
|
|
ret = qemuMonitorSetPassword(priv->mon, type, password, connected);
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
|
|
|
|
if (ret == -2) {
|
|
|
|
if (type != VIR_DOMAIN_GRAPHICS_TYPE_VNC) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Graphics password only supported for VNC"));
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
ret = -1;
|
|
|
|
} else {
|
2016-06-28 12:39:58 +00:00
|
|
|
ret = qemuMonitorSetVNCPassword(priv->mon, password);
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
}
|
|
|
|
}
|
2011-01-18 18:37:45 +00:00
|
|
|
if (ret != 0)
|
2013-01-10 21:03:14 +00:00
|
|
|
goto end_job;
|
2011-01-18 18:37:45 +00:00
|
|
|
|
2016-06-29 15:14:52 +00:00
|
|
|
if (password[0] == '\0' ||
|
|
|
|
(auth->expires && auth->validTo <= now)) {
|
|
|
|
expire = "now";
|
2016-06-28 12:39:58 +00:00
|
|
|
} else if (auth->expires) {
|
2018-04-25 12:42:34 +00:00
|
|
|
if (virAsprintf(&validTo, "%lu", (unsigned long)auth->validTo) < 0)
|
2016-06-29 15:14:52 +00:00
|
|
|
goto end_job;
|
|
|
|
expire = validTo;
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
} else {
|
2016-06-29 15:14:52 +00:00
|
|
|
expire = "never";
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
}
|
|
|
|
|
2016-06-29 15:14:52 +00:00
|
|
|
ret = qemuMonitorExpirePassword(priv->mon, type, expire);
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
|
|
|
|
if (ret == -2) {
|
|
|
|
/* XXX we could fake this with a timer */
|
|
|
|
if (auth->expires) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Expiry of passwords is not supported"));
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
ret = -1;
|
2011-01-18 18:37:45 +00:00
|
|
|
} else {
|
|
|
|
ret = 0;
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
end_job:
|
2015-01-07 12:12:18 +00:00
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
ret = -1;
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2016-06-29 15:14:52 +00:00
|
|
|
VIR_FREE(validTo);
|
2013-01-10 21:03:14 +00:00
|
|
|
virObjectUnref(cfg);
|
Use the new set_password monitor command to set password.
We try to use that command first when setting a VNC/SPICE password. If
that doesn't work we fallback to the legacy VNC only password
Allow an expiry time to be set, if that doesn't work, throw an error
if they try to use SPICE.
Change since v1:
- moved qemuInitGraphicsPasswords to qemu_hotplug, renamed
to qemuDomainChangeGraphicsPasswords.
- updated what looks like a typo (that appears to work anyway) in
initial patch from Daniel:
- ret = qemuInitGraphicsPasswords(driver, vm,
- VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
- &vm->def->graphics[0]->data.vnc.auth,
- driver->vncPassword);
+ ret = qemuInitGraphicsPasswords(driver, vm,
+ VIR_DOMAIN_GRAPHICS_TYPE_SPICE,
+ &vm->def->graphics[0]->data.spice.auth,
+ driver->spicePassword);
Based on patch by Daniel P. Berrange <berrange@redhat.com>.
2011-01-10 11:12:33 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2011-05-18 16:20:53 +00:00
|
|
|
|
2012-11-28 16:43:10 +00:00
|
|
|
int qemuDomainAttachLease(virQEMUDriverPtr driver,
|
2011-05-18 16:20:53 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainLeaseDefPtr lease)
|
|
|
|
{
|
2013-01-10 21:03:14 +00:00
|
|
|
int ret = -1;
|
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
|
|
|
|
2011-05-18 16:20:53 +00:00
|
|
|
if (virDomainLeaseInsertPreAlloc(vm->def) < 0)
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2011-05-18 16:20:53 +00:00
|
|
|
|
2013-01-10 21:03:14 +00:00
|
|
|
if (virDomainLockLeaseAttach(driver->lockManager, cfg->uri,
|
2012-09-17 13:36:47 +00:00
|
|
|
vm, lease) < 0) {
|
2011-05-18 16:20:53 +00:00
|
|
|
virDomainLeaseInsertPreAlloced(vm->def, NULL);
|
2013-01-10 21:03:14 +00:00
|
|
|
goto cleanup;
|
2011-05-18 16:20:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virDomainLeaseInsertPreAlloced(vm->def, lease);
|
2013-01-10 21:03:14 +00:00
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2013-01-10 21:03:14 +00:00
|
|
|
virObjectUnref(cfg);
|
|
|
|
return ret;
|
2011-05-18 16:20:53 +00:00
|
|
|
}
|
|
|
|
|
2012-11-28 16:43:10 +00:00
|
|
|
int qemuDomainDetachLease(virQEMUDriverPtr driver,
|
2011-05-18 16:20:53 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainLeaseDefPtr lease)
|
|
|
|
{
|
util: eliminate device object leaks related to virDomain*Remove*()
There are several functions in domain_conf.c that remove a device
object from the domain's list of that object type, but don't free the
object or return it to the caller to free. In many cases this isn't a
problem because the caller already had a pointer to the object and
frees it afterward, but in several cases the removed object was just
left floating around with no references to it.
In particular, the function qemuDomainDetachDeviceConfig() calls
functions to locate and remove net (virDomainNetRemoveByMac), disk
(virDomainDiskRemoveByName()), and lease (virDomainLeaseRemove())
devices, but neither it nor its caller qemuDomainModifyDeviceConfig()
ever obtain a pointer to the device being removed, much less free it.
This patch modifies the following "remove" functions to return a
pointer to the device object being removed from the domain device
arrays, to give the caller the option of freeing the device object
using that pointer if needed. In places where the object was
previously leaked, it is now freed:
virDomainDiskRemove
virDomainDiskRemoveByName
virDomainNetRemove
virDomainNetRemoveByMac
virDomainHostdevRemove
virDomainLeaseRemove
virDomainLeaseRemoveAt
The functions that had been leaking:
libxlDomainDetachConfig - leaked a virDomainDiskDef
qemuDomainDetachDeviceConfig - could leak a virDomainDiskDef,
a virDomainNetDef, or a
virDomainLeaseDef
qemuDomainDetachLease - leaked a virDomainLeaseDef
2012-03-06 23:06:14 +00:00
|
|
|
virDomainLeaseDefPtr det_lease;
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
int idx;
|
2011-05-18 16:20:53 +00:00
|
|
|
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
if ((idx = virDomainLeaseIndex(vm->def, lease)) < 0) {
|
2012-07-18 15:22:03 +00:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("Lease %s in lockspace %s does not exist"),
|
|
|
|
lease->key, NULLSTR(lease->lockspace));
|
2011-05-18 16:20:53 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virDomainLockLeaseDetach(driver->lockManager, vm, lease) < 0)
|
|
|
|
return -1;
|
|
|
|
|
Convert 'int i' to 'size_t i' in src/qemu 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 14:09:33 +00:00
|
|
|
det_lease = virDomainLeaseRemoveAt(vm->def, idx);
|
util: eliminate device object leaks related to virDomain*Remove*()
There are several functions in domain_conf.c that remove a device
object from the domain's list of that object type, but don't free the
object or return it to the caller to free. In many cases this isn't a
problem because the caller already had a pointer to the object and
frees it afterward, but in several cases the removed object was just
left floating around with no references to it.
In particular, the function qemuDomainDetachDeviceConfig() calls
functions to locate and remove net (virDomainNetRemoveByMac), disk
(virDomainDiskRemoveByName()), and lease (virDomainLeaseRemove())
devices, but neither it nor its caller qemuDomainModifyDeviceConfig()
ever obtain a pointer to the device being removed, much less free it.
This patch modifies the following "remove" functions to return a
pointer to the device object being removed from the domain device
arrays, to give the caller the option of freeing the device object
using that pointer if needed. In places where the object was
previously leaked, it is now freed:
virDomainDiskRemove
virDomainDiskRemoveByName
virDomainNetRemove
virDomainNetRemoveByMac
virDomainHostdevRemove
virDomainLeaseRemove
virDomainLeaseRemoveAt
The functions that had been leaking:
libxlDomainDetachConfig - leaked a virDomainDiskDef
qemuDomainDetachDeviceConfig - could leak a virDomainDiskDef,
a virDomainNetDef, or a
virDomainLeaseDef
qemuDomainDetachLease - leaked a virDomainLeaseDef
2012-03-06 23:06:14 +00:00
|
|
|
virDomainLeaseDefFree(det_lease);
|
2011-05-18 16:20:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2013-03-13 10:08:55 +00:00
|
|
|
|
|
|
|
int qemuDomainDetachChrDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainChrDefPtr chr,
|
|
|
|
bool async)
|
2013-03-13 10:08:55 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virDomainDefPtr vmdef = vm->def;
|
|
|
|
virDomainChrDefPtr tmpChr;
|
2019-02-11 13:16:58 +00:00
|
|
|
bool guestfwd = false;
|
2013-03-13 10:08:55 +00:00
|
|
|
|
|
|
|
if (!(tmpChr = virDomainChrFind(vmdef, chr))) {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2018-01-23 04:24:42 +00:00
|
|
|
_("chr type '%s' device not present "
|
|
|
|
"in domain configuration"),
|
|
|
|
virDomainChrDeviceTypeToString(chr->deviceType));
|
2016-06-13 16:30:34 +00:00
|
|
|
goto cleanup;
|
2013-03-13 10:08:55 +00:00
|
|
|
}
|
|
|
|
|
2019-02-11 13:16:58 +00:00
|
|
|
/* guestfwd channels are not really -device rather than
|
|
|
|
* -netdev. We need to treat them slightly differently. */
|
|
|
|
guestfwd = tmpChr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CHANNEL &&
|
|
|
|
tmpChr->targetType == VIR_DOMAIN_CHR_CHANNEL_TARGET_TYPE_GUESTFWD;
|
|
|
|
|
|
|
|
if (!async && !guestfwd)
|
2018-05-23 16:01:16 +00:00
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &tmpChr->info);
|
2013-07-11 15:11:02 +00:00
|
|
|
|
2019-02-11 13:16:58 +00:00
|
|
|
if (guestfwd) {
|
2019-03-12 12:49:24 +00:00
|
|
|
int rc;
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
rc = qemuMonitorRemoveNetdev(priv->mon, tmpChr->info.alias);
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
rc = -1;
|
|
|
|
|
|
|
|
if (rc < 0)
|
2019-02-11 13:16:58 +00:00
|
|
|
goto cleanup;
|
|
|
|
} else {
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, tmpChr->info.alias) < 0)
|
2019-02-11 13:16:58 +00:00
|
|
|
goto cleanup;
|
2016-10-24 19:47:56 +00:00
|
|
|
}
|
2013-03-13 10:08:55 +00:00
|
|
|
|
2019-02-11 13:16:58 +00:00
|
|
|
if (guestfwd) {
|
|
|
|
ret = qemuDomainRemoveChrDevice(driver, vm, tmpChr, false);
|
|
|
|
} else if (async) {
|
2018-05-23 16:01:16 +00:00
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
2019-02-11 13:16:58 +00:00
|
|
|
ret = qemuDomainRemoveChrDevice(driver, vm, tmpChr, true);
|
2018-05-23 16:01:16 +00:00
|
|
|
}
|
2015-03-02 09:59:52 +00:00
|
|
|
|
2014-03-25 06:49:44 +00:00
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2013-03-13 10:08:55 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2015-01-17 05:09:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainDetachRNGDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainRNGDefPtr rng,
|
|
|
|
bool async)
|
2015-01-17 05:09:38 +00:00
|
|
|
{
|
|
|
|
ssize_t idx;
|
|
|
|
virDomainRNGDefPtr tmpRNG;
|
|
|
|
int ret = -1;
|
|
|
|
|
2017-09-05 08:06:33 +00:00
|
|
|
if ((idx = virDomainRNGFind(vm->def, rng)) < 0) {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2018-01-23 04:24:42 +00:00
|
|
|
_("model '%s' RNG device not present "
|
|
|
|
"in domain configuration"),
|
|
|
|
virDomainRNGBackendTypeToString(rng->model));
|
2015-01-17 05:09:38 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
tmpRNG = vm->def->rngs[idx];
|
|
|
|
|
|
|
|
if (!tmpRNG->info.alias) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("alias not set for RNG device"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &tmpRNG->info);
|
2015-01-17 05:09:38 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, tmpRNG->info.alias) < 0)
|
2015-01-17 05:09:38 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveRNGDevice(driver, vm, tmpRNG);
|
|
|
|
}
|
2015-01-17 05:09:38 +00:00
|
|
|
|
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2015-01-17 05:09:38 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2015-01-21 16:45:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainDetachMemoryDevice(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainMemoryDefPtr memdef,
|
|
|
|
bool async)
|
2015-01-21 16:45:54 +00:00
|
|
|
{
|
|
|
|
virDomainMemoryDefPtr mem;
|
|
|
|
int idx;
|
|
|
|
int ret = -1;
|
|
|
|
|
2015-07-31 14:00:20 +00:00
|
|
|
qemuDomainMemoryDeviceAlignSize(vm->def, memdef);
|
2015-01-21 16:45:54 +00:00
|
|
|
|
|
|
|
if ((idx = virDomainMemoryFindByDef(vm->def, memdef)) < 0) {
|
2018-01-23 04:24:44 +00:00
|
|
|
virReportError(VIR_ERR_DEVICE_MISSING,
|
2018-01-23 04:24:42 +00:00
|
|
|
_("model '%s' memory device not present "
|
|
|
|
"in the domain configuration"),
|
|
|
|
virDomainMemoryModelTypeToString(memdef->model));
|
2015-01-21 16:45:54 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
mem = vm->def->mems[idx];
|
|
|
|
|
|
|
|
if (!mem->info.alias) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("alias for the memory device was not found"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &mem->info);
|
2015-01-21 16:45:54 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, mem->info.alias) < 0)
|
2015-01-21 16:45:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveMemoryDevice(driver, vm, mem);
|
|
|
|
}
|
2015-01-21 16:45:54 +00:00
|
|
|
|
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2015-01-21 16:45:54 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2016-08-16 13:02:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuDomainRemoveVcpu(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
unsigned int vcpu)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
virDomainVcpuDefPtr vcpuinfo = virDomainDefGetVcpu(vm->def, vcpu);
|
|
|
|
qemuDomainVcpuPrivatePtr vcpupriv = QEMU_DOMAIN_VCPU_PRIVATE(vcpuinfo);
|
|
|
|
int oldvcpus = virDomainDefGetVcpus(vm->def);
|
|
|
|
unsigned int nvcpus = vcpupriv->vcpus;
|
2017-09-25 20:34:44 +00:00
|
|
|
virErrorPtr save_error = NULL;
|
2016-08-16 13:02:11 +00:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (qemuDomainRefreshVcpuInfo(driver, vm, QEMU_ASYNC_JOB_NONE, false) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* validation requires us to set the expected state prior to calling it */
|
|
|
|
for (i = vcpu; i < vcpu + nvcpus; i++) {
|
|
|
|
vcpuinfo = virDomainDefGetVcpu(vm->def, i);
|
|
|
|
vcpuinfo->online = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (qemuDomainValidateVcpuInfo(vm) < 0) {
|
|
|
|
/* rollback vcpu count if the setting has failed */
|
|
|
|
virDomainAuditVcpu(vm, oldvcpus, oldvcpus - nvcpus, "update", false);
|
|
|
|
|
|
|
|
for (i = vcpu; i < vcpu + nvcpus; i++) {
|
|
|
|
vcpuinfo = virDomainDefGetVcpu(vm->def, i);
|
|
|
|
vcpuinfo->online = true;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainAuditVcpu(vm, oldvcpus, oldvcpus - nvcpus, "update", true);
|
|
|
|
|
2017-09-25 20:34:44 +00:00
|
|
|
virErrorPreserveLast(&save_error);
|
|
|
|
|
|
|
|
for (i = vcpu; i < vcpu + nvcpus; i++)
|
|
|
|
ignore_value(virCgroupDelThread(priv->cgroup, VIR_CGROUP_THREAD_VCPU, i));
|
|
|
|
|
|
|
|
virErrorRestore(&save_error);
|
2016-08-16 13:02:11 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
qemuDomainRemoveVcpuAlias(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
const char *alias)
|
|
|
|
{
|
|
|
|
virDomainVcpuDefPtr vcpu;
|
|
|
|
qemuDomainVcpuPrivatePtr vcpupriv;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < virDomainDefGetVcpusMax(vm->def); i++) {
|
|
|
|
vcpu = virDomainDefGetVcpu(vm->def, i);
|
|
|
|
vcpupriv = QEMU_DOMAIN_VCPU_PRIVATE(vcpu);
|
|
|
|
|
|
|
|
if (STREQ_NULLABLE(alias, vcpupriv->alias)) {
|
|
|
|
qemuDomainRemoveVcpu(driver, vm, i);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-13 12:12:34 +00:00
|
|
|
static int
|
2016-08-16 13:02:11 +00:00
|
|
|
qemuDomainHotplugDelVcpu(virQEMUDriverPtr driver,
|
2017-04-13 12:22:16 +00:00
|
|
|
virQEMUDriverConfigPtr cfg,
|
2016-08-16 13:02:11 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
unsigned int vcpu)
|
|
|
|
{
|
|
|
|
virDomainVcpuDefPtr vcpuinfo = virDomainDefGetVcpu(vm->def, vcpu);
|
|
|
|
qemuDomainVcpuPrivatePtr vcpupriv = QEMU_DOMAIN_VCPU_PRIVATE(vcpuinfo);
|
|
|
|
int oldvcpus = virDomainDefGetVcpus(vm->def);
|
|
|
|
unsigned int nvcpus = vcpupriv->vcpus;
|
|
|
|
int rc;
|
2017-03-03 15:04:57 +00:00
|
|
|
int ret = -1;
|
2016-08-16 13:02:11 +00:00
|
|
|
|
|
|
|
if (!vcpupriv->alias) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("vcpu '%u' can't be unplugged"), vcpu);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuDomainMarkDeviceAliasForRemoval(vm, vcpupriv->alias);
|
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, vcpupriv->alias) < 0) {
|
|
|
|
if (virDomainObjIsActive(vm))
|
|
|
|
virDomainAuditVcpu(vm, oldvcpus, oldvcpus - nvcpus, "update", false);
|
2017-03-03 15:04:57 +00:00
|
|
|
goto cleanup;
|
2016-08-16 13:02:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if ((rc = qemuDomainWaitForDeviceRemoval(vm)) <= 0) {
|
|
|
|
if (rc == 0)
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("vcpu unplug request timed out"));
|
|
|
|
|
2017-03-03 15:04:57 +00:00
|
|
|
goto cleanup;
|
2016-08-16 13:02:11 +00:00
|
|
|
}
|
|
|
|
|
2017-03-03 15:04:57 +00:00
|
|
|
if (qemuDomainRemoveVcpu(driver, vm, vcpu) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2017-04-13 12:22:16 +00:00
|
|
|
qemuDomainVcpuPersistOrder(vm->def);
|
|
|
|
|
|
|
|
if (virDomainSaveStatus(driver->xmlopt, cfg->stateDir, vm, driver->caps) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2017-03-03 15:04:57 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
|
|
|
return ret;
|
2016-08-16 13:02:11 +00:00
|
|
|
}
|
2016-11-24 15:56:56 +00:00
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuDomainHotplugAddVcpu(virQEMUDriverPtr driver,
|
2017-04-13 12:22:16 +00:00
|
|
|
virQEMUDriverConfigPtr cfg,
|
2016-11-24 15:56:56 +00:00
|
|
|
virDomainObjPtr vm,
|
|
|
|
unsigned int vcpu)
|
|
|
|
{
|
|
|
|
virJSONValuePtr vcpuprops = NULL;
|
|
|
|
virDomainVcpuDefPtr vcpuinfo = virDomainDefGetVcpu(vm->def, vcpu);
|
|
|
|
qemuDomainVcpuPrivatePtr vcpupriv = QEMU_DOMAIN_VCPU_PRIVATE(vcpuinfo);
|
|
|
|
unsigned int nvcpus = vcpupriv->vcpus;
|
|
|
|
bool newhotplug = qemuDomainSupportsNewVcpuHotplug(vm);
|
|
|
|
int ret = -1;
|
|
|
|
int rc;
|
|
|
|
int oldvcpus = virDomainDefGetVcpus(vm->def);
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (newhotplug) {
|
|
|
|
if (virAsprintf(&vcpupriv->alias, "vcpu%u", vcpu) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!(vcpuprops = qemuBuildHotpluggableCPUProps(vcpuinfo)))
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
qemuDomainObjEnterMonitor(driver, vm);
|
|
|
|
|
|
|
|
if (newhotplug) {
|
|
|
|
rc = qemuMonitorAddDeviceArgs(qemuDomainGetMonitor(vm), vcpuprops);
|
|
|
|
vcpuprops = NULL;
|
|
|
|
} else {
|
|
|
|
rc = qemuMonitorSetCPU(qemuDomainGetMonitor(vm), vcpu, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
virDomainAuditVcpu(vm, oldvcpus, oldvcpus + nvcpus, "update", rc == 0);
|
|
|
|
|
|
|
|
if (rc < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* start outputting of the new XML element to allow keeping unpluggability */
|
|
|
|
if (newhotplug)
|
|
|
|
vm->def->individualvcpus = true;
|
|
|
|
|
|
|
|
if (qemuDomainRefreshVcpuInfo(driver, vm, QEMU_ASYNC_JOB_NONE, false) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* validation requires us to set the expected state prior to calling it */
|
|
|
|
for (i = vcpu; i < vcpu + nvcpus; i++) {
|
|
|
|
vcpuinfo = virDomainDefGetVcpu(vm->def, i);
|
|
|
|
vcpupriv = QEMU_DOMAIN_VCPU_PRIVATE(vcpuinfo);
|
|
|
|
|
|
|
|
vcpuinfo->online = true;
|
|
|
|
|
|
|
|
if (vcpupriv->tid > 0 &&
|
|
|
|
qemuProcessSetupVcpu(vm, i) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (qemuDomainValidateVcpuInfo(vm) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2017-04-13 12:22:16 +00:00
|
|
|
qemuDomainVcpuPersistOrder(vm->def);
|
|
|
|
|
|
|
|
if (virDomainSaveStatus(driver->xmlopt, cfg->stateDir, vm, driver->caps) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2016-11-24 15:56:56 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virJSONValueFree(vcpuprops);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qemuDomainSelectHotplugVcpuEntities:
|
|
|
|
*
|
|
|
|
* @def: domain definition
|
|
|
|
* @nvcpus: target vcpu count
|
|
|
|
* @enable: set to true if vcpus should be enabled
|
|
|
|
*
|
|
|
|
* Tries to find which vcpu entities need to be enabled or disabled to reach
|
|
|
|
* @nvcpus. This function works in order of the legacy hotplug but is able to
|
|
|
|
* skip over entries that are added out of order.
|
|
|
|
*
|
|
|
|
* Returns the bitmap of vcpus to modify on success, NULL on error.
|
|
|
|
*/
|
|
|
|
static virBitmapPtr
|
|
|
|
qemuDomainSelectHotplugVcpuEntities(virDomainDefPtr def,
|
|
|
|
unsigned int nvcpus,
|
|
|
|
bool *enable)
|
|
|
|
{
|
|
|
|
virBitmapPtr ret = NULL;
|
|
|
|
virDomainVcpuDefPtr vcpu;
|
|
|
|
qemuDomainVcpuPrivatePtr vcpupriv;
|
|
|
|
unsigned int maxvcpus = virDomainDefGetVcpusMax(def);
|
|
|
|
unsigned int curvcpus = virDomainDefGetVcpus(def);
|
|
|
|
ssize_t i;
|
|
|
|
|
|
|
|
if (!(ret = virBitmapNew(maxvcpus)))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (nvcpus > curvcpus) {
|
|
|
|
*enable = true;
|
|
|
|
|
|
|
|
for (i = 0; i < maxvcpus && curvcpus < nvcpus; i++) {
|
|
|
|
vcpu = virDomainDefGetVcpu(def, i);
|
|
|
|
vcpupriv = QEMU_DOMAIN_VCPU_PRIVATE(vcpu);
|
|
|
|
|
|
|
|
if (vcpu->online)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (vcpupriv->vcpus == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
curvcpus += vcpupriv->vcpus;
|
|
|
|
|
|
|
|
if (curvcpus > nvcpus) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("target vm vcpu granularity does not allow the "
|
|
|
|
"desired vcpu count"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ignore_value(virBitmapSetBit(ret, i));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
*enable = false;
|
|
|
|
|
|
|
|
for (i = maxvcpus - 1; i >= 0 && curvcpus > nvcpus; i--) {
|
|
|
|
vcpu = virDomainDefGetVcpu(def, i);
|
|
|
|
vcpupriv = QEMU_DOMAIN_VCPU_PRIVATE(vcpu);
|
|
|
|
|
|
|
|
if (!vcpu->online)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (vcpupriv->vcpus == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!vcpupriv->alias)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
curvcpus -= vcpupriv->vcpus;
|
|
|
|
|
|
|
|
if (curvcpus < nvcpus) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("target vm vcpu granularity does not allow the "
|
|
|
|
"desired vcpu count"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ignore_value(virBitmapSetBit(ret, i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (curvcpus != nvcpus) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("failed to find appropriate hotpluggable vcpus to "
|
|
|
|
"reach the desired target vcpu count"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virBitmapFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
qemuDomainSetVcpusLive(virQEMUDriverPtr driver,
|
|
|
|
virQEMUDriverConfigPtr cfg,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virBitmapPtr vcpumap,
|
|
|
|
bool enable)
|
|
|
|
{
|
|
|
|
qemuDomainObjPrivatePtr priv = vm->privateData;
|
|
|
|
qemuCgroupEmulatorAllNodesDataPtr emulatorCgroup = NULL;
|
|
|
|
ssize_t nextvcpu = -1;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (qemuCgroupEmulatorAllNodesAllow(priv->cgroup, &emulatorCgroup) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (enable) {
|
|
|
|
while ((nextvcpu = virBitmapNextSetBit(vcpumap, nextvcpu)) != -1) {
|
2017-04-13 12:22:16 +00:00
|
|
|
if (qemuDomainHotplugAddVcpu(driver, cfg, vm, nextvcpu) < 0)
|
|
|
|
goto cleanup;
|
2016-11-24 15:56:56 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (nextvcpu = virDomainDefGetVcpusMax(vm->def) - 1; nextvcpu >= 0; nextvcpu--) {
|
|
|
|
if (!virBitmapIsBitSet(vcpumap, nextvcpu))
|
|
|
|
continue;
|
|
|
|
|
2017-04-13 12:22:16 +00:00
|
|
|
if (qemuDomainHotplugDelVcpu(driver, cfg, vm, nextvcpu) < 0)
|
|
|
|
goto cleanup;
|
2016-11-24 15:56:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
qemuCgroupEmulatorAllNodesRestore(emulatorCgroup);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qemuDomainSetVcpusConfig:
|
|
|
|
* @def: config/offline definition of a domain
|
|
|
|
* @nvcpus: target vcpu count
|
|
|
|
*
|
|
|
|
* Properly handle cold(un)plug of vcpus:
|
|
|
|
* - plug in inactive vcpus/uplug active rather than rewriting state
|
|
|
|
* - fix hotpluggable state
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
qemuDomainSetVcpusConfig(virDomainDefPtr def,
|
|
|
|
unsigned int nvcpus,
|
|
|
|
bool hotpluggable)
|
|
|
|
{
|
|
|
|
virDomainVcpuDefPtr vcpu;
|
|
|
|
size_t curvcpus = virDomainDefGetVcpus(def);
|
|
|
|
size_t maxvcpus = virDomainDefGetVcpusMax(def);
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
/* ordering information may become invalid, thus clear it */
|
|
|
|
virDomainDefVcpuOrderClear(def);
|
|
|
|
|
|
|
|
if (curvcpus == nvcpus)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (curvcpus < nvcpus) {
|
|
|
|
for (i = 0; i < maxvcpus; i++) {
|
|
|
|
vcpu = virDomainDefGetVcpu(def, i);
|
|
|
|
|
|
|
|
if (!vcpu)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (vcpu->online) {
|
2018-12-04 17:08:14 +00:00
|
|
|
/* non-hotpluggable vcpus need to be clustered at the beginning,
|
2016-11-24 15:56:56 +00:00
|
|
|
* thus we need to force vcpus to be hotpluggable when we find
|
|
|
|
* vcpus that are hotpluggable and online prior to the ones
|
|
|
|
* we are going to add */
|
|
|
|
if (vcpu->hotpluggable == VIR_TRISTATE_BOOL_YES)
|
|
|
|
hotpluggable = true;
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
vcpu->online = true;
|
|
|
|
if (hotpluggable) {
|
|
|
|
vcpu->hotpluggable = VIR_TRISTATE_BOOL_YES;
|
|
|
|
def->individualvcpus = true;
|
|
|
|
} else {
|
|
|
|
vcpu->hotpluggable = VIR_TRISTATE_BOOL_NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (++curvcpus == nvcpus)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (i = maxvcpus; i != 0; i--) {
|
|
|
|
vcpu = virDomainDefGetVcpu(def, i - 1);
|
|
|
|
|
|
|
|
if (!vcpu || !vcpu->online)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
vcpu->online = false;
|
|
|
|
vcpu->hotpluggable = VIR_TRISTATE_BOOL_YES;
|
|
|
|
|
|
|
|
if (--curvcpus == nvcpus)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainSetVcpusInternal(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
virDomainDefPtr persistentDef,
|
|
|
|
unsigned int nvcpus,
|
|
|
|
bool hotpluggable)
|
|
|
|
{
|
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
|
|
|
virBitmapPtr vcpumap = NULL;
|
|
|
|
bool enable;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (def && nvcpus > virDomainDefGetVcpusMax(def)) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("requested vcpus is greater than max allowable"
|
|
|
|
" vcpus for the live domain: %u > %u"),
|
|
|
|
nvcpus, virDomainDefGetVcpusMax(def));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (persistentDef && nvcpus > virDomainDefGetVcpusMax(persistentDef)) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("requested vcpus is greater than max allowable"
|
|
|
|
" vcpus for the persistent domain: %u > %u"),
|
|
|
|
nvcpus, virDomainDefGetVcpusMax(persistentDef));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def) {
|
|
|
|
if (!(vcpumap = qemuDomainSelectHotplugVcpuEntities(vm->def, nvcpus,
|
|
|
|
&enable)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (qemuDomainSetVcpusLive(driver, cfg, vm, vcpumap, enable) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (persistentDef) {
|
|
|
|
qemuDomainSetVcpusConfig(persistentDef, nvcpus, hotpluggable);
|
|
|
|
|
|
|
|
if (virDomainSaveConfig(cfg->configDir, driver->caps, persistentDef) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virBitmapFree(vcpumap);
|
|
|
|
virObjectUnref(cfg);
|
|
|
|
return ret;
|
|
|
|
}
|
2016-06-21 15:17:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemuDomainSetVcpuConfig(virDomainDefPtr def,
|
|
|
|
virBitmapPtr map,
|
|
|
|
bool state)
|
|
|
|
{
|
|
|
|
virDomainVcpuDefPtr vcpu;
|
|
|
|
ssize_t next = -1;
|
|
|
|
|
|
|
|
def->individualvcpus = true;
|
|
|
|
|
2017-03-31 11:05:47 +00:00
|
|
|
/* ordering information may become invalid, thus clear it */
|
|
|
|
virDomainDefVcpuOrderClear(def);
|
|
|
|
|
2017-03-31 11:02:14 +00:00
|
|
|
while ((next = virBitmapNextSetBit(map, next)) >= 0) {
|
2016-06-21 15:17:41 +00:00
|
|
|
if (!(vcpu = virDomainDefGetVcpu(def, next)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
vcpu->online = state;
|
|
|
|
vcpu->hotpluggable = VIR_TRISTATE_BOOL_YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* qemuDomainFilterHotplugVcpuEntities:
|
|
|
|
*
|
|
|
|
* Returns a bitmap of hotpluggable vcpu entities that correspond to the logical
|
|
|
|
* vcpus requested in @vcpus.
|
|
|
|
*/
|
|
|
|
static virBitmapPtr
|
|
|
|
qemuDomainFilterHotplugVcpuEntities(virDomainDefPtr def,
|
|
|
|
virBitmapPtr vcpus,
|
|
|
|
bool state)
|
|
|
|
{
|
|
|
|
qemuDomainVcpuPrivatePtr vcpupriv;
|
|
|
|
virDomainVcpuDefPtr vcpu;
|
|
|
|
virBitmapPtr map = NULL;
|
|
|
|
virBitmapPtr ret = NULL;
|
|
|
|
ssize_t next = -1;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!(map = virBitmapNewCopy(vcpus)))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* make sure that all selected vcpus are in the correct state */
|
2017-03-31 11:02:14 +00:00
|
|
|
while ((next = virBitmapNextSetBit(map, next)) >= 0) {
|
2016-06-21 15:17:41 +00:00
|
|
|
if (!(vcpu = virDomainDefGetVcpu(def, next)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (vcpu->online == state) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
2017-03-31 11:28:19 +00:00
|
|
|
_("vcpu '%zd' is already in requested state"), next);
|
2016-06-21 15:17:41 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vcpu->online && !vcpu->hotpluggable) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
2017-03-31 11:28:19 +00:00
|
|
|
_("vcpu '%zd' can't be hotunplugged"), next);
|
2016-06-21 15:17:41 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make sure that all vCPUs belonging to a single hotpluggable entity were
|
|
|
|
* selected and then de-select any sub-threads of it. */
|
|
|
|
next = -1;
|
2017-03-31 11:02:14 +00:00
|
|
|
while ((next = virBitmapNextSetBit(map, next)) >= 0) {
|
2016-06-21 15:17:41 +00:00
|
|
|
if (!(vcpu = virDomainDefGetVcpu(def, next)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
vcpupriv = QEMU_DOMAIN_VCPU_PRIVATE(vcpu);
|
|
|
|
|
|
|
|
if (vcpupriv->vcpus == 0) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
2017-03-31 11:28:19 +00:00
|
|
|
_("vcpu '%zd' belongs to a larger hotpluggable entity, "
|
2016-06-21 15:17:41 +00:00
|
|
|
"but siblings were not selected"), next);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = next + 1; i < next + vcpupriv->vcpus; i++) {
|
|
|
|
if (!virBitmapIsBitSet(map, i)) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("vcpu '%zu' was not selected but it belongs to "
|
2017-03-31 11:28:19 +00:00
|
|
|
"hotpluggable entity '%zd-%zd' which was "
|
2016-06-21 15:17:41 +00:00
|
|
|
"partially selected"),
|
|
|
|
i, next, next + vcpupriv->vcpus - 1);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* clear the subthreads */
|
|
|
|
ignore_value(virBitmapClearBit(map, i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_STEAL_PTR(ret, map);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virBitmapFree(map);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-03-31 11:13:14 +00:00
|
|
|
static int
|
2017-03-31 11:34:16 +00:00
|
|
|
qemuDomainVcpuValidateConfig(virDomainDefPtr def,
|
2017-06-28 08:42:49 +00:00
|
|
|
virBitmapPtr map)
|
2017-03-31 11:13:14 +00:00
|
|
|
{
|
2017-03-31 11:34:16 +00:00
|
|
|
virDomainVcpuDefPtr vcpu;
|
|
|
|
size_t maxvcpus = virDomainDefGetVcpusMax(def);
|
|
|
|
ssize_t next;
|
2017-05-12 15:46:31 +00:00
|
|
|
ssize_t firstvcpu = -1;
|
2017-03-31 11:34:16 +00:00
|
|
|
|
2017-06-28 08:42:49 +00:00
|
|
|
/* vcpu 0 can't be modified */
|
|
|
|
if (virBitmapIsBitSet(map, 0)) {
|
2017-03-31 11:13:14 +00:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG, "%s",
|
2017-06-28 08:42:49 +00:00
|
|
|
_("vCPU '0' can't be modified"));
|
2017-03-31 11:13:14 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-03-31 11:34:16 +00:00
|
|
|
/* non-hotpluggable vcpus need to stay clustered starting from vcpu 0 */
|
|
|
|
for (next = virBitmapNextSetBit(map, -1) + 1; next < maxvcpus; next++) {
|
|
|
|
if (!(vcpu = virDomainDefGetVcpu(def, next)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* skip vcpus being modified */
|
2017-05-12 15:46:31 +00:00
|
|
|
if (virBitmapIsBitSet(map, next)) {
|
|
|
|
if (firstvcpu < 0)
|
|
|
|
firstvcpu = next;
|
|
|
|
|
2017-03-31 11:34:16 +00:00
|
|
|
continue;
|
2017-05-12 15:46:31 +00:00
|
|
|
}
|
2017-03-31 11:34:16 +00:00
|
|
|
|
|
|
|
if (vcpu->online && vcpu->hotpluggable == VIR_TRISTATE_BOOL_NO) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("vcpu '%zd' can't be modified as it is followed "
|
2017-05-12 15:46:31 +00:00
|
|
|
"by non-hotpluggable online vcpus"), firstvcpu);
|
2017-03-31 11:34:16 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-31 11:13:14 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-21 15:17:41 +00:00
|
|
|
int
|
|
|
|
qemuDomainSetVcpuInternal(virQEMUDriverPtr driver,
|
|
|
|
virDomainObjPtr vm,
|
|
|
|
virDomainDefPtr def,
|
|
|
|
virDomainDefPtr persistentDef,
|
|
|
|
virBitmapPtr map,
|
|
|
|
bool state)
|
|
|
|
{
|
|
|
|
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
|
|
|
virBitmapPtr livevcpus = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (def) {
|
|
|
|
if (!qemuDomainSupportsNewVcpuHotplug(vm)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("this qemu version does not support specific "
|
|
|
|
"vCPU hotplug"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(livevcpus = qemuDomainFilterHotplugVcpuEntities(def, map, state)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* Make sure that only one hotpluggable entity is selected.
|
|
|
|
* qemuDomainSetVcpusLive allows setting more at once but error
|
|
|
|
* resolution in case of a partial failure is hard, so don't let users
|
|
|
|
* do so */
|
|
|
|
if (virBitmapCountBits(livevcpus) != 1) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
|
|
|
_("only one hotpluggable entity can be selected"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-31 11:13:14 +00:00
|
|
|
if (persistentDef) {
|
2017-06-28 08:42:49 +00:00
|
|
|
if (qemuDomainVcpuValidateConfig(persistentDef, map) < 0)
|
2017-03-31 11:13:14 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2016-06-21 15:17:41 +00:00
|
|
|
if (livevcpus &&
|
|
|
|
qemuDomainSetVcpusLive(driver, cfg, vm, livevcpus, state) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (persistentDef) {
|
|
|
|
qemuDomainSetVcpuConfig(persistentDef, map, state);
|
|
|
|
|
|
|
|
if (virDomainSaveConfig(cfg->configDir, driver->caps, persistentDef) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virBitmapFree(livevcpus);
|
|
|
|
virObjectUnref(cfg);
|
|
|
|
return ret;
|
|
|
|
}
|
2017-10-17 14:44:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainDetachInputDevice(virDomainObjPtr vm,
|
2018-05-23 16:01:16 +00:00
|
|
|
virDomainInputDefPtr def,
|
|
|
|
bool async)
|
2017-10-17 14:44:24 +00:00
|
|
|
{
|
|
|
|
virDomainInputDefPtr input;
|
|
|
|
int ret = -1;
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
if ((idx = virDomainInputDefFind(vm->def, def)) < 0) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("matching input device not found"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
input = vm->def->inputs[idx];
|
|
|
|
|
|
|
|
switch ((virDomainInputBus) input->bus) {
|
|
|
|
case VIR_DOMAIN_INPUT_BUS_PS2:
|
|
|
|
case VIR_DOMAIN_INPUT_BUS_XEN:
|
|
|
|
case VIR_DOMAIN_INPUT_BUS_PARALLELS:
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
|
|
|
|
_("input device on bus '%s' cannot be detached"),
|
|
|
|
virDomainInputBusTypeToString(input->bus));
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
case VIR_DOMAIN_INPUT_BUS_LAST:
|
|
|
|
case VIR_DOMAIN_INPUT_BUS_USB:
|
|
|
|
case VIR_DOMAIN_INPUT_BUS_VIRTIO:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &input->info);
|
2017-10-17 14:44:24 +00:00
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, input->info.alias) < 0)
|
2017-10-17 14:44:24 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2018-05-23 16:01:16 +00:00
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveInputDevice(vm, input);
|
|
|
|
}
|
2017-10-17 14:44:24 +00:00
|
|
|
|
|
|
|
cleanup:
|
2018-05-23 16:01:16 +00:00
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
2017-10-17 14:44:24 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2018-05-30 10:49:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
qemuDomainDetachVsockDevice(virDomainObjPtr vm,
|
|
|
|
virDomainVsockDefPtr dev,
|
|
|
|
bool async)
|
|
|
|
{
|
|
|
|
virDomainVsockDefPtr vsock = vm->def->vsock;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
|
|
|
|
if (!vsock ||
|
|
|
|
!virDomainVsockDefEquals(dev, vsock)) {
|
|
|
|
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("matching vsock device not found"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!async)
|
|
|
|
qemuDomainMarkDeviceForRemoval(vm, &vsock->info);
|
|
|
|
|
2019-03-12 12:49:24 +00:00
|
|
|
if (qemuDomainDeleteDevice(vm, vsock->info.alias) < 0)
|
2018-05-30 10:49:04 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (async) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
if ((ret = qemuDomainWaitForDeviceRemoval(vm)) == 1)
|
|
|
|
ret = qemuDomainRemoveVsockDevice(vm, vsock);
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (!async)
|
|
|
|
qemuDomainResetDeviceRemoval(vm);
|
|
|
|
return ret;
|
|
|
|
}
|