mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-03 11:35:19 +00:00
qemu: Add secret object hotplug for TCP chardev TLS
https://bugzilla.redhat.com/show_bug.cgi?id=1300776 Complete the implementation of support for TLS encryption on chardev TCP transports by adding the hotplug ability of a secret to generate the passwordid for the TLS object for chrdev, RNG, and redirdev. Fix up the order of object removal on failure to be the inverse of the attempted attach (for redirdev, chr, rng) - for each the tls object was being removed before the chardev backend. Likewise, add the ability to hot unplug that secret object as well and be sure the order of unplug matches that inverse order of plug. Signed-off-by: John Ferlan <jferlan@redhat.com>
This commit is contained in:
parent
daf5c651f0
commit
8550e8585e
@ -7567,7 +7567,7 @@ qemuDomainAttachDeviceLive(virDomainObjPtr vm,
|
||||
break;
|
||||
|
||||
case VIR_DOMAIN_DEVICE_REDIRDEV:
|
||||
ret = qemuDomainAttachRedirdevDevice(driver, vm,
|
||||
ret = qemuDomainAttachRedirdevDevice(conn, driver, vm,
|
||||
dev->data.redirdev);
|
||||
if (!ret) {
|
||||
alias = dev->data.redirdev->info.alias;
|
||||
@ -7576,7 +7576,7 @@ qemuDomainAttachDeviceLive(virDomainObjPtr vm,
|
||||
break;
|
||||
|
||||
case VIR_DOMAIN_DEVICE_CHR:
|
||||
ret = qemuDomainAttachChrDevice(driver, vm,
|
||||
ret = qemuDomainAttachChrDevice(conn, driver, vm,
|
||||
dev->data.chr);
|
||||
if (!ret) {
|
||||
alias = dev->data.chr->info.alias;
|
||||
@ -7585,7 +7585,7 @@ qemuDomainAttachDeviceLive(virDomainObjPtr vm,
|
||||
break;
|
||||
|
||||
case VIR_DOMAIN_DEVICE_RNG:
|
||||
ret = qemuDomainAttachRNGDevice(driver, vm,
|
||||
ret = qemuDomainAttachRNGDevice(conn, driver, vm,
|
||||
dev->data.rng);
|
||||
if (!ret) {
|
||||
alias = dev->data.rng->info.alias;
|
||||
|
@ -1480,16 +1480,31 @@ qemuDomainGetChardevTLSObjects(virQEMUDriverConfigPtr cfg,
|
||||
virDomainChrSourceDefPtr dev,
|
||||
char *charAlias,
|
||||
virJSONValuePtr *tlsProps,
|
||||
char **tlsAlias)
|
||||
char **tlsAlias,
|
||||
virJSONValuePtr *secProps,
|
||||
char **secAlias)
|
||||
{
|
||||
qemuDomainChrSourcePrivatePtr chrSourcePriv =
|
||||
QEMU_DOMAIN_CHR_SOURCE_PRIVATE(dev);
|
||||
|
||||
if (dev->type != VIR_DOMAIN_CHR_TYPE_TCP ||
|
||||
dev->data.tcp.haveTLS != VIR_TRISTATE_BOOL_YES)
|
||||
return 0;
|
||||
|
||||
/* Add a secret object in order to access the TLS environment.
|
||||
* The secinfo will only be created for serial TCP device. */
|
||||
if (chrSourcePriv && chrSourcePriv->secinfo) {
|
||||
if (qemuBuildSecretInfoProps(chrSourcePriv->secinfo, secProps) < 0)
|
||||
return -1;
|
||||
|
||||
if (!(*secAlias = qemuDomainGetSecretAESAlias(charAlias, false)))
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (qemuBuildTLSx509BackendProps(cfg->chardevTLSx509certdir,
|
||||
dev->data.tcp.listen,
|
||||
cfg->chardevTLSx509verify,
|
||||
NULL,
|
||||
*secAlias,
|
||||
priv->qemuCaps,
|
||||
tlsProps) < 0)
|
||||
return -1;
|
||||
@ -1502,7 +1517,8 @@ qemuDomainGetChardevTLSObjects(virQEMUDriverConfigPtr cfg,
|
||||
}
|
||||
|
||||
|
||||
int qemuDomainAttachRedirdevDevice(virQEMUDriverPtr driver,
|
||||
int qemuDomainAttachRedirdevDevice(virConnectPtr conn,
|
||||
virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainRedirdevDefPtr redirdev)
|
||||
{
|
||||
@ -1515,8 +1531,11 @@ int qemuDomainAttachRedirdevDevice(virQEMUDriverPtr driver,
|
||||
char *devstr = NULL;
|
||||
bool chardevAdded = false;
|
||||
bool tlsobjAdded = false;
|
||||
bool secobjAdded = false;
|
||||
virJSONValuePtr tlsProps = NULL;
|
||||
virJSONValuePtr secProps = NULL;
|
||||
char *tlsAlias = NULL;
|
||||
char *secAlias = NULL;
|
||||
virErrorPtr orig_err;
|
||||
|
||||
qemuDomainPrepareChardevSourceTLS(redirdev->source, cfg);
|
||||
@ -1533,11 +1552,26 @@ int qemuDomainAttachRedirdevDevice(virQEMUDriverPtr driver,
|
||||
if (VIR_REALLOC_N(def->redirdevs, def->nredirdevs+1) < 0)
|
||||
goto cleanup;
|
||||
|
||||
if (qemuDomainSecretChardevPrepare(conn, cfg, priv, redirdev->info.alias,
|
||||
redirdev->source) < 0)
|
||||
goto cleanup;
|
||||
|
||||
if (qemuDomainGetChardevTLSObjects(cfg, priv, redirdev->source,
|
||||
charAlias, &tlsProps, &tlsAlias) < 0)
|
||||
charAlias, &tlsProps, &tlsAlias,
|
||||
&secProps, &secAlias) < 0)
|
||||
goto cleanup;
|
||||
|
||||
qemuDomainObjEnterMonitor(driver, vm);
|
||||
|
||||
if (secAlias) {
|
||||
rc = qemuMonitorAddObject(priv->mon, "secret",
|
||||
secAlias, secProps);
|
||||
secProps = NULL;
|
||||
if (rc < 0)
|
||||
goto exit_monitor;
|
||||
secobjAdded = true;
|
||||
}
|
||||
|
||||
if (tlsAlias) {
|
||||
rc = qemuMonitorAddObject(priv->mon, "tls-creds-x509",
|
||||
tlsAlias, tlsProps);
|
||||
@ -1566,6 +1600,8 @@ int qemuDomainAttachRedirdevDevice(virQEMUDriverPtr driver,
|
||||
cleanup:
|
||||
VIR_FREE(tlsAlias);
|
||||
virJSONValueFree(tlsProps);
|
||||
VIR_FREE(secAlias);
|
||||
virJSONValueFree(secProps);
|
||||
VIR_FREE(charAlias);
|
||||
VIR_FREE(devstr);
|
||||
virObjectUnref(cfg);
|
||||
@ -1573,11 +1609,13 @@ int qemuDomainAttachRedirdevDevice(virQEMUDriverPtr driver,
|
||||
|
||||
exit_monitor:
|
||||
orig_err = virSaveLastError();
|
||||
if (tlsobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
||||
/* detach associated chardev on error */
|
||||
if (chardevAdded)
|
||||
ignore_value(qemuMonitorDetachCharDev(priv->mon, charAlias));
|
||||
if (tlsobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
||||
if (secobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, secAlias));
|
||||
if (orig_err) {
|
||||
virSetError(orig_err);
|
||||
virFreeError(orig_err);
|
||||
@ -1753,7 +1791,8 @@ qemuDomainAttachChrDeviceAssignAddr(virDomainObjPtr vm,
|
||||
return ret;
|
||||
}
|
||||
|
||||
int qemuDomainAttachChrDevice(virQEMUDriverPtr driver,
|
||||
int qemuDomainAttachChrDevice(virConnectPtr conn,
|
||||
virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainChrDefPtr chr)
|
||||
{
|
||||
@ -1767,8 +1806,11 @@ int qemuDomainAttachChrDevice(virQEMUDriverPtr driver,
|
||||
char *charAlias = NULL;
|
||||
bool chardevAttached = false;
|
||||
bool tlsobjAdded = false;
|
||||
bool secobjAdded = false;
|
||||
virJSONValuePtr tlsProps = NULL;
|
||||
char *tlsAlias = NULL;
|
||||
virJSONValuePtr secProps = NULL;
|
||||
char *secAlias = NULL;
|
||||
bool need_release = false;
|
||||
|
||||
if (chr->deviceType == VIR_DOMAIN_CHR_DEVICE_TYPE_CHANNEL &&
|
||||
@ -1794,11 +1836,25 @@ int qemuDomainAttachChrDevice(virQEMUDriverPtr driver,
|
||||
if (qemuDomainChrPreInsert(vmdef, chr) < 0)
|
||||
goto cleanup;
|
||||
|
||||
if (qemuDomainSecretChardevPrepare(conn, cfg, priv, chr->info.alias,
|
||||
dev) < 0)
|
||||
goto cleanup;
|
||||
|
||||
if (qemuDomainGetChardevTLSObjects(cfg, priv, dev, charAlias,
|
||||
&tlsProps, &tlsAlias) < 0)
|
||||
&tlsProps, &tlsAlias,
|
||||
&secProps, &secAlias) < 0)
|
||||
goto cleanup;
|
||||
|
||||
qemuDomainObjEnterMonitor(driver, vm);
|
||||
if (secAlias) {
|
||||
rc = qemuMonitorAddObject(priv->mon, "secret",
|
||||
secAlias, secProps);
|
||||
secProps = NULL;
|
||||
if (rc < 0)
|
||||
goto exit_monitor;
|
||||
secobjAdded = true;
|
||||
}
|
||||
|
||||
if (tlsAlias) {
|
||||
rc = qemuMonitorAddObject(priv->mon, "tls-creds-x509",
|
||||
tlsAlias, tlsProps);
|
||||
@ -1829,6 +1885,8 @@ int qemuDomainAttachChrDevice(virQEMUDriverPtr driver,
|
||||
qemuDomainReleaseDeviceAddress(vm, &chr->info, NULL);
|
||||
VIR_FREE(tlsAlias);
|
||||
virJSONValueFree(tlsProps);
|
||||
VIR_FREE(secAlias);
|
||||
virJSONValueFree(secProps);
|
||||
VIR_FREE(charAlias);
|
||||
VIR_FREE(devstr);
|
||||
virObjectUnref(cfg);
|
||||
@ -1836,11 +1894,13 @@ int qemuDomainAttachChrDevice(virQEMUDriverPtr driver,
|
||||
|
||||
exit_monitor:
|
||||
orig_err = virSaveLastError();
|
||||
if (tlsobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
||||
/* detach associated chardev on error */
|
||||
if (chardevAttached)
|
||||
qemuMonitorDetachCharDev(priv->mon, charAlias);
|
||||
if (tlsobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
||||
if (secobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, secAlias));
|
||||
if (orig_err) {
|
||||
virSetError(orig_err);
|
||||
virFreeError(orig_err);
|
||||
@ -1852,7 +1912,8 @@ int qemuDomainAttachChrDevice(virQEMUDriverPtr driver,
|
||||
|
||||
|
||||
int
|
||||
qemuDomainAttachRNGDevice(virQEMUDriverPtr driver,
|
||||
qemuDomainAttachRNGDevice(virConnectPtr conn,
|
||||
virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainRNGDefPtr rng)
|
||||
{
|
||||
@ -1863,12 +1924,15 @@ qemuDomainAttachRNGDevice(virQEMUDriverPtr driver,
|
||||
char *charAlias = NULL;
|
||||
char *objAlias = NULL;
|
||||
char *tlsAlias = NULL;
|
||||
char *secAlias = NULL;
|
||||
bool releaseaddr = false;
|
||||
bool chardevAdded = false;
|
||||
bool objAdded = false;
|
||||
bool tlsobjAdded = false;
|
||||
bool secobjAdded = false;
|
||||
virJSONValuePtr props = NULL;
|
||||
virJSONValuePtr tlsProps = NULL;
|
||||
virJSONValuePtr secProps = NULL;
|
||||
virDomainCCWAddressSetPtr ccwaddrs = NULL;
|
||||
const char *type;
|
||||
int ret = -1;
|
||||
@ -1923,13 +1987,28 @@ qemuDomainAttachRNGDevice(virQEMUDriverPtr driver,
|
||||
if (!(charAlias = qemuAliasChardevFromDevAlias(rng->info.alias)))
|
||||
goto cleanup;
|
||||
|
||||
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD &&
|
||||
qemuDomainGetChardevTLSObjects(cfg, priv, rng->source.chardev,
|
||||
charAlias, &tlsProps, &tlsAlias) < 0)
|
||||
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD) {
|
||||
if (qemuDomainSecretChardevPrepare(conn, cfg, priv, rng->info.alias,
|
||||
rng->source.chardev) < 0)
|
||||
goto cleanup;
|
||||
|
||||
if (qemuDomainGetChardevTLSObjects(cfg, priv, rng->source.chardev,
|
||||
charAlias, &tlsProps, &tlsAlias,
|
||||
&secProps, &secAlias) < 0)
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
qemuDomainObjEnterMonitor(driver, vm);
|
||||
|
||||
if (secAlias) {
|
||||
rv = qemuMonitorAddObject(priv->mon, "secret",
|
||||
secAlias, secProps);
|
||||
secProps = NULL;
|
||||
if (rv < 0)
|
||||
goto exit_monitor;
|
||||
secobjAdded = true;
|
||||
}
|
||||
|
||||
if (tlsAlias) {
|
||||
rv = qemuMonitorAddObject(priv->mon, "tls-creds-x509",
|
||||
tlsAlias, tlsProps);
|
||||
@ -1967,10 +2046,12 @@ qemuDomainAttachRNGDevice(virQEMUDriverPtr driver,
|
||||
virDomainAuditRNG(vm, NULL, rng, "attach", ret == 0);
|
||||
cleanup:
|
||||
virJSONValueFree(tlsProps);
|
||||
virJSONValueFree(secProps);
|
||||
virJSONValueFree(props);
|
||||
if (ret < 0 && releaseaddr)
|
||||
qemuDomainReleaseDeviceAddress(vm, &rng->info, NULL);
|
||||
VIR_FREE(tlsAlias);
|
||||
VIR_FREE(secAlias);
|
||||
VIR_FREE(charAlias);
|
||||
VIR_FREE(objAlias);
|
||||
VIR_FREE(devstr);
|
||||
@ -1980,12 +2061,14 @@ qemuDomainAttachRNGDevice(virQEMUDriverPtr driver,
|
||||
|
||||
exit_monitor:
|
||||
orig_err = virSaveLastError();
|
||||
if (tlsobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
||||
if (objAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, objAlias));
|
||||
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD && chardevAdded)
|
||||
ignore_value(qemuMonitorDetachCharDev(priv->mon, charAlias));
|
||||
if (tlsobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
||||
if (secobjAdded)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, secAlias));
|
||||
if (orig_err) {
|
||||
virSetError(orig_err);
|
||||
virFreeError(orig_err);
|
||||
@ -3554,6 +3637,7 @@ qemuDomainRemoveChrDevice(virQEMUDriverPtr driver,
|
||||
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
||||
char *charAlias = NULL;
|
||||
char *tlsAlias = NULL;
|
||||
char *secAlias = NULL;
|
||||
qemuDomainObjPrivatePtr priv = vm->privateData;
|
||||
int ret = -1;
|
||||
int rc;
|
||||
@ -3565,15 +3649,29 @@ qemuDomainRemoveChrDevice(virQEMUDriverPtr driver,
|
||||
goto cleanup;
|
||||
|
||||
if (chr->source->type == VIR_DOMAIN_CHR_TYPE_TCP &&
|
||||
chr->source->data.tcp.haveTLS == VIR_TRISTATE_BOOL_YES &&
|
||||
!(tlsAlias = qemuAliasTLSObjFromChardevAlias(charAlias)))
|
||||
chr->source->data.tcp.haveTLS == VIR_TRISTATE_BOOL_YES) {
|
||||
|
||||
if (!(tlsAlias = qemuAliasTLSObjFromChardevAlias(charAlias)))
|
||||
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(charAlias, false)))
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
qemuDomainObjEnterMonitor(driver, vm);
|
||||
rc = qemuMonitorDetachCharDev(priv->mon, charAlias);
|
||||
|
||||
if (tlsAlias && qemuMonitorDelObject(priv->mon, tlsAlias) < 0)
|
||||
goto exit_monitor;
|
||||
if (rc == 0) {
|
||||
if (tlsAlias)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
||||
if (secAlias)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, secAlias));
|
||||
}
|
||||
|
||||
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
||||
goto cleanup;
|
||||
@ -3593,12 +3691,9 @@ qemuDomainRemoveChrDevice(virQEMUDriverPtr driver,
|
||||
cleanup:
|
||||
VIR_FREE(charAlias);
|
||||
VIR_FREE(tlsAlias);
|
||||
VIR_FREE(secAlias);
|
||||
virObjectUnref(cfg);
|
||||
return ret;
|
||||
|
||||
exit_monitor:
|
||||
ignore_value(qemuDomainObjExitMonitor(driver, vm));
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
|
||||
@ -3608,9 +3703,11 @@ qemuDomainRemoveRNGDevice(virQEMUDriverPtr driver,
|
||||
virDomainRNGDefPtr rng)
|
||||
{
|
||||
virObjectEventPtr event;
|
||||
virQEMUDriverConfigPtr cfg = virQEMUDriverGetConfig(driver);
|
||||
char *charAlias = NULL;
|
||||
char *objAlias = NULL;
|
||||
char *tlsAlias = NULL;
|
||||
char *secAlias = NULL;
|
||||
qemuDomainObjPrivatePtr priv = vm->privateData;
|
||||
ssize_t idx;
|
||||
int ret = -1;
|
||||
@ -3626,10 +3723,19 @@ qemuDomainRemoveRNGDevice(virQEMUDriverPtr driver,
|
||||
if (!(charAlias = qemuAliasChardevFromDevAlias(rng->info.alias)))
|
||||
goto cleanup;
|
||||
|
||||
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD &&
|
||||
!(tlsAlias = qemuAliasTLSObjFromChardevAlias(charAlias)))
|
||||
if (rng->backend == VIR_DOMAIN_RNG_BACKEND_EGD) {
|
||||
if (!(tlsAlias = qemuAliasTLSObjFromChardevAlias(charAlias)))
|
||||
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(charAlias, false)))
|
||||
goto cleanup;
|
||||
}
|
||||
|
||||
qemuDomainObjEnterMonitor(driver, vm);
|
||||
|
||||
rc = qemuMonitorDelObject(priv->mon, objAlias);
|
||||
@ -3638,6 +3744,8 @@ qemuDomainRemoveRNGDevice(virQEMUDriverPtr driver,
|
||||
ignore_value(qemuMonitorDetachCharDev(priv->mon, charAlias));
|
||||
if (tlsAlias)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, tlsAlias));
|
||||
if (secAlias)
|
||||
ignore_value(qemuMonitorDelObject(priv->mon, secAlias));
|
||||
}
|
||||
|
||||
if (qemuDomainObjExitMonitor(driver, vm) < 0)
|
||||
@ -3661,6 +3769,8 @@ qemuDomainRemoveRNGDevice(virQEMUDriverPtr driver,
|
||||
VIR_FREE(charAlias);
|
||||
VIR_FREE(objAlias);
|
||||
VIR_FREE(tlsAlias);
|
||||
VIR_FREE(secAlias);
|
||||
virObjectUnref(cfg);
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
@ -43,7 +43,8 @@ int qemuDomainAttachDeviceDiskLive(virConnectPtr conn,
|
||||
int qemuDomainAttachNetDevice(virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainNetDefPtr net);
|
||||
int qemuDomainAttachRedirdevDevice(virQEMUDriverPtr driver,
|
||||
int qemuDomainAttachRedirdevDevice(virConnectPtr conn,
|
||||
virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainRedirdevDefPtr hostdev);
|
||||
int qemuDomainAttachHostDevice(virConnectPtr conn,
|
||||
@ -92,13 +93,15 @@ int qemuDomainAttachLease(virQEMUDriverPtr driver,
|
||||
int qemuDomainDetachLease(virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainLeaseDefPtr lease);
|
||||
int qemuDomainAttachChrDevice(virQEMUDriverPtr driver,
|
||||
int qemuDomainAttachChrDevice(virConnectPtr conn,
|
||||
virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainChrDefPtr chr);
|
||||
int qemuDomainDetachChrDevice(virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainChrDefPtr chr);
|
||||
int qemuDomainAttachRNGDevice(virQEMUDriverPtr driver,
|
||||
int qemuDomainAttachRNGDevice(virConnectPtr conn,
|
||||
virQEMUDriverPtr driver,
|
||||
virDomainObjPtr vm,
|
||||
virDomainRNGDefPtr rng);
|
||||
int qemuDomainDetachRNGDevice(virQEMUDriverPtr driver,
|
||||
|
@ -118,7 +118,7 @@ testQemuHotplugAttach(virDomainObjPtr vm,
|
||||
ret = qemuDomainAttachDeviceDiskLive(NULL, &driver, vm, dev);
|
||||
break;
|
||||
case VIR_DOMAIN_DEVICE_CHR:
|
||||
ret = qemuDomainAttachChrDevice(&driver, vm, dev->data.chr);
|
||||
ret = qemuDomainAttachChrDevice(NULL, &driver, vm, dev->data.chr);
|
||||
break;
|
||||
default:
|
||||
VIR_TEST_VERBOSE("device type '%s' cannot be attached\n",
|
||||
|
Loading…
Reference in New Issue
Block a user