2009-04-17 16:09:07 +00:00
|
|
|
/** @file vbox_tmpl.c
|
|
|
|
* Template File to support multiple versions of VirtualBox
|
|
|
|
* at runtime :).
|
|
|
|
*
|
|
|
|
* IMPORTANT:
|
|
|
|
* Please dont include this file in the src/Makefile.am, it
|
|
|
|
* is automatically include by other files.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
* Copyright (C) 2010-2011 Red Hat, Inc.
|
2009-04-17 16:09:07 +00:00
|
|
|
* Copyright (C) 2008-2009 Sun Microsystems, Inc.
|
|
|
|
*
|
|
|
|
* This file is part of a free software library; you can redistribute
|
|
|
|
* it and/or modify it under the terms of the GNU Lesser General
|
|
|
|
* Public License version 2.1 as published by the Free Software
|
|
|
|
* Foundation and shipped in the "COPYING" file with this library.
|
|
|
|
* The library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY of any kind.
|
|
|
|
*
|
|
|
|
* Sun LGPL Disclaimer: For the avoidance of doubt, except that if
|
|
|
|
* any license choice other than GPL or LGPL is available it will
|
|
|
|
* apply instead, Sun elects to use only the Lesser General Public
|
|
|
|
* License version 2.1 (LGPLv2) at this time for any software where
|
|
|
|
* a choice of LGPL license versions is made available with the
|
|
|
|
* language indicating that LGPLv2 or any later version may be used,
|
|
|
|
* or where a choice of which version of the LGPL is applied is
|
|
|
|
* otherwise unspecified.
|
|
|
|
*
|
|
|
|
* Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
|
|
|
|
* Clara, CA 95054 USA or visit http://www.sun.com if you need
|
|
|
|
* additional information or have any questions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <sys/utsname.h>
|
2010-12-14 22:07:57 +00:00
|
|
|
#include <unistd.h>
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
#include "internal.h"
|
|
|
|
#include "datatypes.h"
|
|
|
|
#include "domain_conf.h"
|
2009-05-06 13:51:19 +00:00
|
|
|
#include "network_conf.h"
|
2009-04-17 16:09:07 +00:00
|
|
|
#include "virterror_internal.h"
|
2009-07-24 16:12:16 +00:00
|
|
|
#include "domain_event.h"
|
2009-09-04 14:28:52 +00:00
|
|
|
#include "storage_conf.h"
|
2009-09-25 13:20:13 +00:00
|
|
|
#include "storage_file.h"
|
2009-04-17 16:09:07 +00:00
|
|
|
#include "uuid.h"
|
2009-07-24 16:12:16 +00:00
|
|
|
#include "event.h"
|
2009-04-17 16:09:07 +00:00
|
|
|
#include "memory.h"
|
|
|
|
#include "nodeinfo.h"
|
|
|
|
#include "logging.h"
|
|
|
|
#include "vbox_driver.h"
|
|
|
|
|
|
|
|
/* This one changes from version to version. */
|
|
|
|
#if VBOX_API_VERSION == 2002
|
|
|
|
# include "vbox_CAPI_v2_2.h"
|
2009-07-24 16:12:16 +00:00
|
|
|
#elif VBOX_API_VERSION == 3000
|
|
|
|
# include "vbox_CAPI_v3_0.h"
|
2009-12-04 13:49:45 +00:00
|
|
|
#elif VBOX_API_VERSION == 3001
|
|
|
|
# include "vbox_CAPI_v3_1.h"
|
2010-05-26 11:54:16 +00:00
|
|
|
#elif VBOX_API_VERSION == 3002
|
|
|
|
# include "vbox_CAPI_v3_2.h"
|
2010-12-27 22:35:30 +00:00
|
|
|
#elif VBOX_API_VERSION == 4000
|
|
|
|
# include "vbox_CAPI_v4_0.h"
|
2009-07-24 16:12:16 +00:00
|
|
|
#else
|
|
|
|
# error "Unsupport VBOX_API_VERSION"
|
2009-04-17 16:09:07 +00:00
|
|
|
#endif
|
|
|
|
|
2009-05-06 13:51:19 +00:00
|
|
|
/* Include this *last* or we'll get the wrong vbox_CAPI_*.h. */
|
2010-12-14 22:07:57 +00:00
|
|
|
#include "vbox_glue.h"
|
2009-05-06 13:51:19 +00:00
|
|
|
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_VBOX
|
|
|
|
#define VBOX_UTF16_FREE(arg) data->pFuncs->pfnUtf16Free(arg)
|
|
|
|
#define VBOX_UTF8_FREE(arg) data->pFuncs->pfnUtf8Free(arg)
|
|
|
|
#define VBOX_COM_UNALLOC_MEM(arg) data->pFuncs->pfnComUnallocMem(arg)
|
|
|
|
#define VBOX_UTF16_TO_UTF8(arg1, arg2) data->pFuncs->pfnUtf16ToUtf8(arg1, arg2)
|
|
|
|
#define VBOX_UTF8_TO_UTF16(arg1, arg2) data->pFuncs->pfnUtf8ToUtf16(arg1, arg2)
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
#define VBOX_ADDREF(arg) (arg)->vtbl->nsisupports.AddRef((nsISupports *)(arg))
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#define VBOX_RELEASE(arg) \
|
|
|
|
do { \
|
|
|
|
if (arg) { \
|
|
|
|
(arg)->vtbl->nsisupports.Release((nsISupports *)(arg)); \
|
|
|
|
(arg) = NULL; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
#define VBOX_OBJECT_CHECK(conn, type, value) \
|
|
|
|
vboxGlobalData *data = conn->privateData;\
|
|
|
|
type ret = value;\
|
|
|
|
if(!data->vboxObj) {\
|
|
|
|
return ret;\
|
|
|
|
}
|
|
|
|
|
|
|
|
#define VBOX_OBJECT_HOST_CHECK(conn, type, value) \
|
|
|
|
vboxGlobalData *data = conn->privateData;\
|
|
|
|
type ret = value;\
|
|
|
|
IHost *host = NULL;\
|
|
|
|
if(!data->vboxObj) {\
|
|
|
|
return ret;\
|
|
|
|
}\
|
|
|
|
data->vboxObj->vtbl->GetHost(data->vboxObj, &host);\
|
|
|
|
if (!host) {\
|
|
|
|
return ret;\
|
|
|
|
}
|
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
|
2010-03-09 18:22:22 +00:00
|
|
|
# define VBOX_MEDIUM_RELEASE(arg) \
|
2009-12-04 13:49:45 +00:00
|
|
|
if(arg)\
|
|
|
|
(arg)->vtbl->imedium.nsisupports.Release((nsISupports *)(arg))
|
2010-03-09 18:22:22 +00:00
|
|
|
# define VBOX_MEDIUM_FUNC_ARG1(object, func, arg1) \
|
2009-12-04 13:49:45 +00:00
|
|
|
(object)->vtbl->imedium.func((IMedium *)(object), arg1)
|
2010-03-09 18:22:22 +00:00
|
|
|
# define VBOX_MEDIUM_FUNC_ARG2(object, func, arg1, arg2) \
|
2009-12-04 13:49:45 +00:00
|
|
|
(object)->vtbl->imedium.func((IMedium *)(object), arg1, arg2)
|
|
|
|
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
|
|
|
|
typedef IMedium IHardDisk;
|
|
|
|
typedef IMediumAttachment IHardDiskAttachment;
|
2010-03-09 18:22:22 +00:00
|
|
|
# define MediaState_Inaccessible MediumState_Inaccessible
|
|
|
|
# define HardDiskVariant_Standard MediumVariant_Standard
|
|
|
|
# define HardDiskVariant_Fixed MediumVariant_Fixed
|
|
|
|
# define VBOX_MEDIUM_RELEASE(arg) VBOX_RELEASE(arg)
|
|
|
|
# define VBOX_MEDIUM_FUNC_ARG1(object, func, arg1) \
|
2009-12-04 13:49:45 +00:00
|
|
|
(object)->vtbl->func(object, arg1)
|
2010-03-09 18:22:22 +00:00
|
|
|
# define VBOX_MEDIUM_FUNC_ARG2(object, func, arg1, arg2) \
|
2009-12-04 13:49:45 +00:00
|
|
|
(object)->vtbl->func(object, arg1, arg2)
|
|
|
|
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-04-02 22:38:57 +00:00
|
|
|
#define vboxError(code, ...) \
|
|
|
|
virReportErrorHelper(NULL, VIR_FROM_VBOX, code, __FILE__, \
|
2010-03-01 23:38:28 +00:00
|
|
|
__FUNCTION__, __LINE__, __VA_ARGS__)
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-07-24 16:12:16 +00:00
|
|
|
#define DEBUGPRUnichar(msg, strUtf16) \
|
|
|
|
if (strUtf16) {\
|
|
|
|
char *strUtf8 = NULL;\
|
|
|
|
\
|
|
|
|
g_pVBoxGlobalData->pFuncs->pfnUtf16ToUtf8(strUtf16, &strUtf8);\
|
|
|
|
if (strUtf8) {\
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("%s: %s", msg, strUtf8);\
|
2009-07-24 16:12:16 +00:00
|
|
|
g_pVBoxGlobalData->pFuncs->pfnUtf8Free(strUtf8);\
|
|
|
|
}\
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DEBUGUUID(msg, iid) \
|
|
|
|
{\
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG (msg ": {%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x}",\
|
2009-07-24 16:12:16 +00:00
|
|
|
(unsigned)(iid)->m0,\
|
|
|
|
(unsigned)(iid)->m1,\
|
|
|
|
(unsigned)(iid)->m2,\
|
|
|
|
(unsigned)(iid)->m3[0],\
|
|
|
|
(unsigned)(iid)->m3[1],\
|
|
|
|
(unsigned)(iid)->m3[2],\
|
|
|
|
(unsigned)(iid)->m3[3],\
|
|
|
|
(unsigned)(iid)->m3[4],\
|
|
|
|
(unsigned)(iid)->m3[5],\
|
|
|
|
(unsigned)(iid)->m3[6],\
|
|
|
|
(unsigned)(iid)->m3[7]);\
|
|
|
|
}\
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
typedef struct {
|
|
|
|
virMutex lock;
|
2010-03-30 14:15:13 +00:00
|
|
|
unsigned long version;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
virCapsPtr caps;
|
|
|
|
|
|
|
|
IVirtualBox *vboxObj;
|
|
|
|
ISession *vboxSession;
|
2009-05-06 13:51:19 +00:00
|
|
|
|
|
|
|
/** Our version specific API table pointer. */
|
|
|
|
PCVBOXXPCOM pFuncs;
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
#if VBOX_API_VERSION == 2002
|
|
|
|
|
|
|
|
} vboxGlobalData;
|
|
|
|
|
|
|
|
#else /* !(VBOX_API_VERSION == 2002) */
|
|
|
|
|
|
|
|
/* An array of callbacks */
|
|
|
|
virDomainEventCallbackListPtr domainEventCallbacks;
|
|
|
|
|
|
|
|
int fdWatch;
|
|
|
|
int domainEventDispatching;
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
# if VBOX_API_VERSION <= 3002
|
|
|
|
/* IVirtualBoxCallback is used in VirtualBox 3.x only */
|
2009-07-24 16:12:16 +00:00
|
|
|
IVirtualBoxCallback *vboxCallback;
|
2010-12-27 22:35:30 +00:00
|
|
|
# endif /* VBOX_API_VERSION <= 3002 */
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
nsIEventQueue *vboxQueue;
|
|
|
|
int volatile vboxCallBackRefCount;
|
|
|
|
|
|
|
|
/* pointer back to the connection */
|
|
|
|
virConnectPtr conn;
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
} vboxGlobalData;
|
|
|
|
|
2009-07-24 16:12:16 +00:00
|
|
|
/* g_pVBoxGlobalData has to be global variable,
|
|
|
|
* there is no other way to make the callbacks
|
|
|
|
* work other then having g_pVBoxGlobalData as
|
|
|
|
* global, because the functions namely AddRef,
|
|
|
|
* Release, etc consider it as global and you
|
|
|
|
* can't change the function definition as it
|
|
|
|
* is XPCOM nsISupport::* function and it expects
|
|
|
|
* them that way
|
|
|
|
*/
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
static vboxGlobalData *g_pVBoxGlobalData = NULL;
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
#endif /* !(VBOX_API_VERSION == 2002) */
|
2009-05-06 13:51:19 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
|
|
|
|
|
|
|
# define VBOX_OBJECT_GET_MACHINE(/* in */ iid_value, /* out */ machine) \
|
|
|
|
data->vboxObj->vtbl->GetMachine(data->vboxObj, iid_value, machine)
|
|
|
|
|
|
|
|
# define VBOX_SESSION_OPEN(/* in */ iid_value, /* unused */ machine) \
|
|
|
|
data->vboxObj->vtbl->OpenSession(data->vboxObj, data->vboxSession, iid_value)
|
|
|
|
|
|
|
|
# define VBOX_SESSION_OPEN_EXISTING(/* in */ iid_value, /* unused */ machine) \
|
|
|
|
data->vboxObj->vtbl->OpenExistingSession(data->vboxObj, data->vboxSession, iid_value)
|
|
|
|
|
|
|
|
# define VBOX_SESSION_CLOSE() \
|
|
|
|
data->vboxSession->vtbl->Close(data->vboxSession)
|
|
|
|
|
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
|
|
|
|
# define VBOX_OBJECT_GET_MACHINE(/* in */ iid_value, /* out */ machine) \
|
|
|
|
data->vboxObj->vtbl->FindMachine(data->vboxObj, iid_value, machine)
|
|
|
|
|
|
|
|
# define VBOX_SESSION_OPEN(/* unused */ iid_value, /* in */ machine) \
|
|
|
|
machine->vtbl->LockMachine(machine, data->vboxSession, LockType_Write)
|
|
|
|
|
|
|
|
# define VBOX_SESSION_OPEN_EXISTING(/* unused */ iid_value, /* in */ machine) \
|
|
|
|
machine->vtbl->LockMachine(machine, data->vboxSession, LockType_Shared)
|
|
|
|
|
|
|
|
# define VBOX_SESSION_CLOSE() \
|
|
|
|
data->vboxSession->vtbl->UnlockMachine(data->vboxSession)
|
|
|
|
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
static virDomainPtr vboxDomainDefineXML(virConnectPtr conn, const char *xml);
|
|
|
|
static int vboxDomainCreate(virDomainPtr dom);
|
|
|
|
static int vboxDomainUndefine(virDomainPtr dom);
|
|
|
|
|
2009-07-24 16:12:16 +00:00
|
|
|
static void vboxDriverLock(vboxGlobalData *data) {
|
|
|
|
virMutexLock(&data->lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vboxDriverUnlock(vboxGlobalData *data) {
|
|
|
|
virMutexUnlock(&data->lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if VBOX_API_VERSION == 2002
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-05-06 13:51:19 +00:00
|
|
|
static void nsIDtoChar(unsigned char *uuid, const nsID *iid) {
|
2009-04-17 16:09:07 +00:00
|
|
|
char uuidstrsrc[VIR_UUID_STRING_BUFLEN];
|
|
|
|
char uuidstrdst[VIR_UUID_STRING_BUFLEN];
|
|
|
|
unsigned char uuidinterim[VIR_UUID_BUFLEN];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
memcpy(uuidinterim, iid, VIR_UUID_BUFLEN);
|
|
|
|
virUUIDFormat(uuidinterim, uuidstrsrc);
|
|
|
|
|
|
|
|
uuidstrdst[0] = uuidstrsrc[6];
|
|
|
|
uuidstrdst[1] = uuidstrsrc[7];
|
|
|
|
uuidstrdst[2] = uuidstrsrc[4];
|
|
|
|
uuidstrdst[3] = uuidstrsrc[5];
|
|
|
|
uuidstrdst[4] = uuidstrsrc[2];
|
|
|
|
uuidstrdst[5] = uuidstrsrc[3];
|
|
|
|
uuidstrdst[6] = uuidstrsrc[0];
|
|
|
|
uuidstrdst[7] = uuidstrsrc[1];
|
|
|
|
|
|
|
|
uuidstrdst[8] = uuidstrsrc[8];
|
|
|
|
|
|
|
|
uuidstrdst[9] = uuidstrsrc[11];
|
|
|
|
uuidstrdst[10] = uuidstrsrc[12];
|
|
|
|
uuidstrdst[11] = uuidstrsrc[9];
|
|
|
|
uuidstrdst[12] = uuidstrsrc[10];
|
|
|
|
|
|
|
|
uuidstrdst[13] = uuidstrsrc[13];
|
|
|
|
|
|
|
|
uuidstrdst[14] = uuidstrsrc[16];
|
|
|
|
uuidstrdst[15] = uuidstrsrc[17];
|
|
|
|
uuidstrdst[16] = uuidstrsrc[14];
|
|
|
|
uuidstrdst[17] = uuidstrsrc[15];
|
|
|
|
|
|
|
|
for(i = 18; i < VIR_UUID_STRING_BUFLEN; i++) {
|
|
|
|
uuidstrdst[i] = uuidstrsrc[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
uuidstrdst[VIR_UUID_STRING_BUFLEN-1] = '\0';
|
|
|
|
virUUIDParse(uuidstrdst, uuid);
|
|
|
|
}
|
|
|
|
|
2009-05-06 13:51:19 +00:00
|
|
|
static void nsIDFromChar(nsID *iid, const unsigned char *uuid) {
|
2009-04-17 16:09:07 +00:00
|
|
|
char uuidstrsrc[VIR_UUID_STRING_BUFLEN];
|
|
|
|
char uuidstrdst[VIR_UUID_STRING_BUFLEN];
|
|
|
|
unsigned char uuidinterim[VIR_UUID_BUFLEN];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
virUUIDFormat(uuid, uuidstrsrc);
|
|
|
|
|
|
|
|
uuidstrdst[0] = uuidstrsrc[6];
|
|
|
|
uuidstrdst[1] = uuidstrsrc[7];
|
|
|
|
uuidstrdst[2] = uuidstrsrc[4];
|
|
|
|
uuidstrdst[3] = uuidstrsrc[5];
|
|
|
|
uuidstrdst[4] = uuidstrsrc[2];
|
|
|
|
uuidstrdst[5] = uuidstrsrc[3];
|
|
|
|
uuidstrdst[6] = uuidstrsrc[0];
|
|
|
|
uuidstrdst[7] = uuidstrsrc[1];
|
|
|
|
|
|
|
|
uuidstrdst[8] = uuidstrsrc[8];
|
|
|
|
|
|
|
|
uuidstrdst[9] = uuidstrsrc[11];
|
|
|
|
uuidstrdst[10] = uuidstrsrc[12];
|
|
|
|
uuidstrdst[11] = uuidstrsrc[9];
|
|
|
|
uuidstrdst[12] = uuidstrsrc[10];
|
|
|
|
|
|
|
|
uuidstrdst[13] = uuidstrsrc[13];
|
|
|
|
|
|
|
|
uuidstrdst[14] = uuidstrsrc[16];
|
|
|
|
uuidstrdst[15] = uuidstrsrc[17];
|
|
|
|
uuidstrdst[16] = uuidstrsrc[14];
|
|
|
|
uuidstrdst[17] = uuidstrsrc[15];
|
|
|
|
|
|
|
|
for(i = 18; i < VIR_UUID_STRING_BUFLEN; i++) {
|
|
|
|
uuidstrdst[i] = uuidstrsrc[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
uuidstrdst[VIR_UUID_STRING_BUFLEN-1] = '\0';
|
|
|
|
virUUIDParse(uuidstrdst, uuidinterim);
|
|
|
|
memcpy(iid, uuidinterim, VIR_UUID_BUFLEN);
|
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
# ifdef WIN32
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
typedef struct _vboxIID_v2_x_WIN32 vboxIID;
|
|
|
|
typedef struct _vboxIID_v2_x_WIN32 vboxIID_v2_x_WIN32;
|
|
|
|
|
|
|
|
struct _vboxIID_v2_x_WIN32 {
|
|
|
|
/* IID is represented by a GUID value. */
|
|
|
|
GUID value;
|
|
|
|
};
|
|
|
|
|
|
|
|
# define VBOX_IID_INITIALIZER { { 0, 0, 0, { 0, 0, 0, 0, 0, 0, 0, 0 } } }
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxIIDUnalloc_v2_x_WIN32(vboxGlobalData *data ATTRIBUTE_UNUSED,
|
|
|
|
vboxIID_v2_x_WIN32 *iid ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
/* Nothing to free */
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static void
|
|
|
|
vboxIIDToUUID_v2_x_WIN32(vboxIID_v2_x_WIN32 *iid, unsigned char *uuid)
|
|
|
|
{
|
|
|
|
nsIDtoChar(uuid, (nsID *)&iid->value);
|
|
|
|
}
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static void
|
|
|
|
vboxIIDFromUUID_v2_x_WIN32(vboxGlobalData *data, vboxIID_v2_x_WIN32 *iid,
|
|
|
|
const unsigned char *uuid)
|
|
|
|
{
|
|
|
|
vboxIIDUnalloc_v2_x_WIN32(data, iid);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
nsIDFromChar((nsID *)&iid->value, uuid);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static bool
|
|
|
|
vboxIIDIsEqual_v2_x_WIN32(vboxIID_v2_x_WIN32 *iid1, vboxIID_v2_x_WIN32 *iid2)
|
|
|
|
{
|
|
|
|
return memcmp(&iid1->value, &iid2->value, sizeof (GUID)) == 0;
|
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static void
|
|
|
|
vboxIIDFromArrayItem_v2_x_WIN32(vboxGlobalData *data, vboxIID_v2_x_WIN32 *iid,
|
|
|
|
vboxArray *array, int idx)
|
|
|
|
{
|
|
|
|
GUID *items = (GUID *)array->items;
|
|
|
|
|
|
|
|
vboxIIDUnalloc_v2_x_WIN32(data, iid);
|
|
|
|
|
|
|
|
memcpy(&iid->value, &items[idx], sizeof (GUID));
|
|
|
|
}
|
|
|
|
|
|
|
|
# define vboxIIDUnalloc(iid) vboxIIDUnalloc_v2_x_WIN32(data, iid)
|
|
|
|
# define vboxIIDToUUID(iid, uuid) vboxIIDToUUID_v2_x_WIN32(iid, uuid)
|
|
|
|
# define vboxIIDFromUUID(iid, uuid) vboxIIDFromUUID_v2_x_WIN32(data, iid, uuid)
|
|
|
|
# define vboxIIDIsEqual(iid1, iid2) vboxIIDIsEqual_v2_x_WIN32(iid1, iid2)
|
|
|
|
# define vboxIIDFromArrayItem(iid, array, idx) \
|
|
|
|
vboxIIDFromArrayItem_v2_x_WIN32(data, iid, array, idx)
|
|
|
|
# define DEBUGIID(msg, iid) DEBUGUUID(msg, (nsID *)&(iid))
|
|
|
|
|
|
|
|
# else /* !WIN32 */
|
|
|
|
|
|
|
|
typedef struct _vboxIID_v2_x vboxIID;
|
|
|
|
typedef struct _vboxIID_v2_x vboxIID_v2_x;
|
|
|
|
|
|
|
|
struct _vboxIID_v2_x {
|
|
|
|
/* IID is represented by a pointer to a nsID. */
|
|
|
|
nsID *value;
|
|
|
|
|
|
|
|
/* backing is used in cases where we need to create or copy an IID.
|
|
|
|
* We cannot allocate memory that can be freed by ComUnallocMem.
|
|
|
|
* Therefore, we use this stack allocated nsID instead. */
|
|
|
|
nsID backing;
|
|
|
|
};
|
|
|
|
|
|
|
|
# define VBOX_IID_INITIALIZER { NULL, { 0, 0, 0, { 0, 0, 0, 0, 0, 0, 0, 0 } } }
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxIIDUnalloc_v2_x(vboxGlobalData *data, vboxIID_v2_x *iid)
|
|
|
|
{
|
|
|
|
if (iid->value == NULL) {
|
2009-09-04 14:28:52 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (iid->value != &iid->backing) {
|
|
|
|
data->pFuncs->pfnComUnallocMem(iid->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
iid->value = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxIIDToUUID_v2_x(vboxIID_v2_x *iid, unsigned char *uuid)
|
|
|
|
{
|
|
|
|
nsIDtoChar(uuid, iid->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxIIDFromUUID_v2_x(vboxGlobalData *data, vboxIID_v2_x *iid,
|
|
|
|
const unsigned char *uuid)
|
|
|
|
{
|
|
|
|
vboxIIDUnalloc_v2_x(data, iid);
|
|
|
|
|
|
|
|
iid->value = &iid->backing;
|
|
|
|
|
|
|
|
nsIDFromChar(iid->value, uuid);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static bool
|
|
|
|
vboxIIDIsEqual_v2_x(vboxIID_v2_x *iid1, vboxIID_v2_x *iid2)
|
|
|
|
{
|
|
|
|
return memcmp(iid1->value, iid2->value, sizeof (nsID)) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxIIDFromArrayItem_v2_x(vboxGlobalData *data, vboxIID_v2_x *iid,
|
|
|
|
vboxArray *array, int idx)
|
|
|
|
{
|
|
|
|
vboxIIDUnalloc_v2_x(data, iid);
|
|
|
|
|
|
|
|
iid->value = &iid->backing;
|
|
|
|
|
|
|
|
memcpy(iid->value, array->items[idx], sizeof (nsID));
|
|
|
|
}
|
|
|
|
|
|
|
|
# define vboxIIDUnalloc(iid) vboxIIDUnalloc_v2_x(data, iid)
|
|
|
|
# define vboxIIDToUUID(iid, uuid) vboxIIDToUUID_v2_x(iid, uuid)
|
|
|
|
# define vboxIIDFromUUID(iid, uuid) vboxIIDFromUUID_v2_x(data, iid, uuid)
|
|
|
|
# define vboxIIDIsEqual(iid1, iid2) vboxIIDIsEqual_v2_x(iid1, iid2)
|
|
|
|
# define vboxIIDFromArrayItem(iid, array, idx) \
|
|
|
|
vboxIIDFromArrayItem_v2_x(data, iid, array, idx)
|
|
|
|
# define DEBUGIID(msg, iid) DEBUGUUID(msg, iid)
|
|
|
|
|
|
|
|
# endif /* !WIN32 */
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
#else /* VBOX_API_VERSION != 2002 */
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
typedef struct _vboxIID_v3_x vboxIID;
|
|
|
|
typedef struct _vboxIID_v3_x vboxIID_v3_x;
|
|
|
|
|
|
|
|
struct _vboxIID_v3_x {
|
|
|
|
/* IID is represented by a UTF-16 encoded UUID in string form. */
|
|
|
|
PRUnichar *value;
|
|
|
|
|
|
|
|
/* owner indicates if we own the value and need to free it. */
|
|
|
|
bool owner;
|
|
|
|
};
|
|
|
|
|
|
|
|
# define VBOX_IID_INITIALIZER { NULL, true }
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxIIDUnalloc_v3_x(vboxGlobalData *data, vboxIID_v3_x *iid)
|
|
|
|
{
|
|
|
|
if (iid->value != NULL && iid->owner) {
|
|
|
|
data->pFuncs->pfnUtf16Free(iid->value);
|
|
|
|
}
|
|
|
|
|
|
|
|
iid->value = NULL;
|
|
|
|
iid->owner = true;
|
2009-07-24 16:12:16 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static void
|
|
|
|
vboxIIDToUUID_v3_x(vboxGlobalData *data, vboxIID_v3_x *iid,
|
|
|
|
unsigned char *uuid)
|
|
|
|
{
|
|
|
|
char *utf8 = NULL;
|
|
|
|
|
|
|
|
data->pFuncs->pfnUtf16ToUtf8(iid->value, &utf8);
|
|
|
|
|
|
|
|
virUUIDParse(utf8, uuid);
|
|
|
|
|
|
|
|
data->pFuncs->pfnUtf8Free(utf8);
|
2009-07-24 16:12:16 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static void
|
|
|
|
vboxIIDFromUUID_v3_x(vboxGlobalData *data, vboxIID_v3_x *iid,
|
|
|
|
const unsigned char *uuid)
|
|
|
|
{
|
|
|
|
char utf8[VIR_UUID_STRING_BUFLEN];
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc_v3_x(data, iid);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
virUUIDFormat(uuid, utf8);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
data->pFuncs->pfnUtf8ToUtf16(utf8, &iid->value);
|
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static bool
|
|
|
|
vboxIIDIsEqual_v3_x(vboxGlobalData *data, vboxIID_v3_x *iid1,
|
|
|
|
vboxIID_v3_x *iid2)
|
|
|
|
{
|
|
|
|
unsigned char uuid1[VIR_UUID_BUFLEN];
|
|
|
|
unsigned char uuid2[VIR_UUID_BUFLEN];
|
2009-09-04 14:28:52 +00:00
|
|
|
|
|
|
|
/* Note: we can't directly compare the utf8 strings here
|
|
|
|
* cause the two UUID's may have seperators as space or '-'
|
|
|
|
* or mixture of both and we don't want to fail here by
|
|
|
|
* using direct string comparison. Here virUUIDParse() takes
|
2010-12-23 16:25:56 +00:00
|
|
|
* care of these cases. */
|
|
|
|
vboxIIDToUUID_v3_x(data, iid1, uuid1);
|
|
|
|
vboxIIDToUUID_v3_x(data, iid2, uuid2);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
return memcmp(uuid1, uuid2, VIR_UUID_BUFLEN) == 0;
|
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static void
|
|
|
|
vboxIIDFromArrayItem_v3_x(vboxGlobalData *data, vboxIID_v3_x *iid,
|
|
|
|
vboxArray *array, int idx)
|
|
|
|
{
|
|
|
|
vboxIIDUnalloc_v3_x(data, iid);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
iid->value = array->items[idx];
|
|
|
|
iid->owner = false;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
|
|
|
|
# define vboxIIDUnalloc(iid) vboxIIDUnalloc_v3_x(data, iid)
|
|
|
|
# define vboxIIDToUUID(iid, uuid) vboxIIDToUUID_v3_x(data, iid, uuid)
|
|
|
|
# define vboxIIDFromUUID(iid, uuid) vboxIIDFromUUID_v3_x(data, iid, uuid)
|
|
|
|
# define vboxIIDIsEqual(iid1, iid2) vboxIIDIsEqual_v3_x(data, iid1, iid2)
|
|
|
|
# define vboxIIDFromArrayItem(iid, array, idx) \
|
|
|
|
vboxIIDFromArrayItem_v3_x(data, iid, array, idx)
|
|
|
|
# define DEBUGIID(msg, strUtf16) DEBUGPRUnichar(msg, strUtf16)
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-03-09 18:22:22 +00:00
|
|
|
# if VBOX_API_VERSION >= 3001
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* function to generate the name for medium,
|
|
|
|
* for e.g: hda, sda, etc
|
|
|
|
*
|
|
|
|
* @returns null terminated string with device name or NULL
|
|
|
|
* for failures
|
|
|
|
* @param conn Input Connection Pointer
|
|
|
|
* @param storageBus Input storage bus type
|
|
|
|
* @param deviceInst Input device instance number
|
|
|
|
* @param devicePort Input port number
|
|
|
|
* @param deviceSlot Input slot number
|
|
|
|
* @param aMaxPortPerInst Input array of max port per device instance
|
|
|
|
* @param aMaxSlotPerPort Input array of max slot per device port
|
|
|
|
*
|
|
|
|
*/
|
2009-12-04 14:18:45 +00:00
|
|
|
static char *vboxGenerateMediumName(PRUint32 storageBus,
|
2009-12-04 13:49:45 +00:00
|
|
|
PRInt32 deviceInst,
|
|
|
|
PRInt32 devicePort,
|
|
|
|
PRInt32 deviceSlot,
|
|
|
|
PRUint32 *aMaxPortPerInst,
|
|
|
|
PRUint32 *aMaxSlotPerPort) {
|
2009-12-04 14:18:45 +00:00
|
|
|
const char *prefix = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
char *name = NULL;
|
|
|
|
int total = 0;
|
|
|
|
PRUint32 maxPortPerInst = 0;
|
|
|
|
PRUint32 maxSlotPerPort = 0;
|
|
|
|
|
|
|
|
if ( !aMaxPortPerInst
|
|
|
|
|| !aMaxSlotPerPort)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if ( (storageBus < StorageBus_IDE)
|
|
|
|
|| (storageBus > StorageBus_Floppy))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
maxPortPerInst = aMaxPortPerInst[storageBus];
|
|
|
|
maxSlotPerPort = aMaxSlotPerPort[storageBus];
|
|
|
|
total = (deviceInst * maxPortPerInst * maxSlotPerPort)
|
|
|
|
+ (devicePort * maxSlotPerPort)
|
|
|
|
+ deviceSlot;
|
|
|
|
|
|
|
|
if (storageBus == StorageBus_IDE) {
|
2009-12-04 14:18:45 +00:00
|
|
|
prefix = "hd";
|
2009-12-04 13:49:45 +00:00
|
|
|
} else if ( (storageBus == StorageBus_SATA)
|
|
|
|
|| (storageBus == StorageBus_SCSI)) {
|
2009-12-04 14:18:45 +00:00
|
|
|
prefix = "sd";
|
2009-12-04 13:49:45 +00:00
|
|
|
} else if (storageBus == StorageBus_Floppy) {
|
2009-12-04 14:18:45 +00:00
|
|
|
prefix = "fd";
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 14:18:45 +00:00
|
|
|
name = virIndexToDiskName(total, prefix);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("name=%s, total=%d, storageBus=%u, deviceInst=%d, "
|
2009-12-04 13:49:45 +00:00
|
|
|
"devicePort=%d deviceSlot=%d, maxPortPerInst=%u maxSlotPerPort=%u",
|
2009-12-04 14:18:45 +00:00
|
|
|
NULLSTR(name), total, storageBus, deviceInst, devicePort,
|
2009-12-04 13:49:45 +00:00
|
|
|
deviceSlot, maxPortPerInst, maxSlotPerPort);
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* function to get the StorageBus, Port number
|
|
|
|
* and Device number for the given devicename
|
|
|
|
* e.g: hda has StorageBus = IDE, port = 0,
|
|
|
|
* device = 0
|
|
|
|
*
|
|
|
|
* @returns true on Success, false on failure.
|
|
|
|
* @param deviceName Input device name
|
|
|
|
* @param aMaxPortPerInst Input array of max port per device instance
|
|
|
|
* @param aMaxSlotPerPort Input array of max slot per device port
|
|
|
|
* @param storageBus Input storage bus type
|
|
|
|
* @param deviceInst Output device instance number
|
|
|
|
* @param devicePort Output port number
|
|
|
|
* @param deviceSlot Output slot number
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static bool vboxGetDeviceDetails(const char *deviceName,
|
|
|
|
PRUint32 *aMaxPortPerInst,
|
|
|
|
PRUint32 *aMaxSlotPerPort,
|
|
|
|
PRUint32 storageBus,
|
|
|
|
PRInt32 *deviceInst,
|
|
|
|
PRInt32 *devicePort,
|
|
|
|
PRInt32 *deviceSlot) {
|
|
|
|
int total = 0;
|
|
|
|
PRUint32 maxPortPerInst = 0;
|
|
|
|
PRUint32 maxSlotPerPort = 0;
|
|
|
|
|
|
|
|
if ( !deviceName
|
|
|
|
|| !deviceInst
|
|
|
|
|| !devicePort
|
|
|
|
|| !deviceSlot
|
|
|
|
|| !aMaxPortPerInst
|
|
|
|
|| !aMaxSlotPerPort)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if ( (storageBus < StorageBus_IDE)
|
|
|
|
|| (storageBus > StorageBus_Floppy))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
total = virDiskNameToIndex(deviceName);
|
|
|
|
|
|
|
|
maxPortPerInst = aMaxPortPerInst[storageBus];
|
|
|
|
maxSlotPerPort = aMaxSlotPerPort[storageBus];
|
|
|
|
|
|
|
|
if ( !maxPortPerInst
|
|
|
|
|| !maxSlotPerPort
|
|
|
|
|| (total < 0))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
*deviceInst = total / (maxPortPerInst * maxSlotPerPort);
|
|
|
|
*devicePort = (total % (maxPortPerInst * maxSlotPerPort)) / maxSlotPerPort;
|
|
|
|
*deviceSlot = (total % (maxPortPerInst * maxSlotPerPort)) % maxSlotPerPort;
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("name=%s, total=%d, storageBus=%u, deviceInst=%d, "
|
2009-12-04 13:49:45 +00:00
|
|
|
"devicePort=%d deviceSlot=%d, maxPortPerInst=%u maxSlotPerPort=%u",
|
|
|
|
deviceName, total, storageBus, *deviceInst, *devicePort,
|
|
|
|
*deviceSlot, maxPortPerInst, maxSlotPerPort);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* function to get the values for max port per
|
|
|
|
* instance and max slots per port for the devices
|
|
|
|
*
|
|
|
|
* @returns true on Success, false on failure.
|
|
|
|
* @param vbox Input IVirtualBox pointer
|
|
|
|
* @param maxPortPerInst Output array of max port per instance
|
|
|
|
* @param maxSlotPerPort Output array of max slot per port
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static bool vboxGetMaxPortSlotValues(IVirtualBox *vbox,
|
|
|
|
PRUint32 *maxPortPerInst,
|
|
|
|
PRUint32 *maxSlotPerPort) {
|
|
|
|
ISystemProperties *sysProps = NULL;
|
|
|
|
|
|
|
|
if (!vbox)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
vbox->vtbl->GetSystemProperties(vbox, &sysProps);
|
|
|
|
|
|
|
|
if (!sysProps)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
sysProps->vtbl->GetMaxPortCountForStorageBus(sysProps,
|
|
|
|
StorageBus_IDE,
|
|
|
|
&maxPortPerInst[StorageBus_IDE]);
|
|
|
|
sysProps->vtbl->GetMaxPortCountForStorageBus(sysProps,
|
|
|
|
StorageBus_SATA,
|
|
|
|
&maxPortPerInst[StorageBus_SATA]);
|
|
|
|
sysProps->vtbl->GetMaxPortCountForStorageBus(sysProps,
|
|
|
|
StorageBus_SCSI,
|
|
|
|
&maxPortPerInst[StorageBus_SCSI]);
|
|
|
|
sysProps->vtbl->GetMaxPortCountForStorageBus(sysProps,
|
|
|
|
StorageBus_Floppy,
|
|
|
|
&maxPortPerInst[StorageBus_Floppy]);
|
|
|
|
|
|
|
|
sysProps->vtbl->GetMaxDevicesPerPortForStorageBus(sysProps,
|
|
|
|
StorageBus_IDE,
|
|
|
|
&maxSlotPerPort[StorageBus_IDE]);
|
|
|
|
sysProps->vtbl->GetMaxDevicesPerPortForStorageBus(sysProps,
|
|
|
|
StorageBus_SATA,
|
|
|
|
&maxSlotPerPort[StorageBus_SATA]);
|
|
|
|
sysProps->vtbl->GetMaxDevicesPerPortForStorageBus(sysProps,
|
|
|
|
StorageBus_SCSI,
|
|
|
|
&maxSlotPerPort[StorageBus_SCSI]);
|
|
|
|
sysProps->vtbl->GetMaxDevicesPerPortForStorageBus(sysProps,
|
|
|
|
StorageBus_Floppy,
|
|
|
|
&maxSlotPerPort[StorageBus_Floppy]);
|
|
|
|
|
|
|
|
VBOX_RELEASE(sysProps);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts Utf-16 string to int
|
|
|
|
*/
|
|
|
|
static int PRUnicharToInt(PRUnichar *strUtf16) {
|
|
|
|
char *strUtf8 = NULL;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (!strUtf16)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
g_pVBoxGlobalData->pFuncs->pfnUtf16ToUtf8(strUtf16, &strUtf8);
|
|
|
|
if (!strUtf8)
|
|
|
|
return -1;
|
|
|
|
|
2010-03-30 14:27:20 +00:00
|
|
|
if (virStrToLong_i(strUtf8, NULL, 10, &ret) < 0)
|
|
|
|
ret = -1;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
g_pVBoxGlobalData->pFuncs->pfnUtf8Free(strUtf8);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts int to Utf-16 string
|
|
|
|
*/
|
|
|
|
static PRUnichar *PRUnicharFromInt(int n) {
|
|
|
|
PRUnichar *strUtf16 = NULL;
|
|
|
|
char s[24];
|
|
|
|
|
|
|
|
snprintf(s, sizeof(s), "%d", n);
|
|
|
|
|
|
|
|
g_pVBoxGlobalData->pFuncs->pfnUtf8ToUtf16(s, &strUtf16);
|
|
|
|
|
|
|
|
return strUtf16;
|
|
|
|
}
|
|
|
|
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif /* VBOX_API_VERSION >= 3001 */
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2009-07-24 16:12:16 +00:00
|
|
|
#endif /* !(VBOX_API_VERSION == 2002) */
|
|
|
|
|
2010-10-22 12:21:31 +00:00
|
|
|
static PRUnichar *
|
|
|
|
vboxSocketFormatAddrUtf16(vboxGlobalData *data, virSocketAddrPtr addr)
|
|
|
|
{
|
|
|
|
char *utf8 = NULL;
|
|
|
|
PRUnichar *utf16 = NULL;
|
|
|
|
|
|
|
|
utf8 = virSocketFormatAddr(addr);
|
|
|
|
|
|
|
|
if (utf8 == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16(utf8, &utf16);
|
|
|
|
VIR_FREE(utf8);
|
|
|
|
|
|
|
|
return utf16;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxSocketParseAddrUtf16(vboxGlobalData *data, const PRUnichar *utf16,
|
|
|
|
virSocketAddrPtr addr)
|
|
|
|
{
|
|
|
|
int result = -1;
|
|
|
|
char *utf8 = NULL;
|
|
|
|
|
|
|
|
VBOX_UTF16_TO_UTF8(utf16, &utf8);
|
|
|
|
|
|
|
|
if (virSocketParseAddr(utf8, addr, AF_UNSPEC) < 0) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_UTF8_FREE(utf8);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
static virCapsPtr vboxCapsInit(void) {
|
|
|
|
struct utsname utsname;
|
|
|
|
virCapsPtr caps;
|
|
|
|
virCapsGuestPtr guest;
|
|
|
|
|
|
|
|
uname(&utsname);
|
|
|
|
|
|
|
|
if ((caps = virCapabilitiesNew(utsname.machine,
|
|
|
|
0, 0)) == NULL)
|
|
|
|
goto no_memory;
|
|
|
|
|
2009-06-03 13:28:02 +00:00
|
|
|
if (nodeCapsInitNUMA(caps) < 0)
|
2009-04-17 16:09:07 +00:00
|
|
|
goto no_memory;
|
|
|
|
|
|
|
|
virCapabilitiesSetMacPrefix(caps, (unsigned char[]){ 0x08, 0x00, 0x27 });
|
|
|
|
|
|
|
|
if ((guest = virCapabilitiesAddGuest(caps,
|
|
|
|
"hvm",
|
|
|
|
utsname.machine,
|
|
|
|
sizeof(void *) * CHAR_BIT,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
NULL)) == NULL)
|
|
|
|
goto no_memory;
|
|
|
|
|
|
|
|
if (virCapabilitiesAddGuestDomain(guest,
|
|
|
|
"vbox",
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
NULL) == NULL)
|
|
|
|
goto no_memory;
|
|
|
|
return caps;
|
|
|
|
|
|
|
|
no_memory:
|
|
|
|
virCapabilitiesFree(caps);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-12-14 22:07:57 +00:00
|
|
|
static int
|
|
|
|
vboxInitialize(vboxGlobalData *data)
|
|
|
|
{
|
2009-05-06 13:51:19 +00:00
|
|
|
data->pFuncs = g_pfnGetFunctions(VBOX_XPCOMC_VERSION);
|
|
|
|
|
|
|
|
if (data->pFuncs == NULL)
|
2009-04-17 16:09:07 +00:00
|
|
|
goto cleanup;
|
2009-05-06 13:51:19 +00:00
|
|
|
|
|
|
|
#if VBOX_XPCOMC_VERSION == 0x00010000U
|
|
|
|
data->pFuncs->pfnComInitialize(&data->vboxObj, &data->vboxSession);
|
2009-07-24 16:12:16 +00:00
|
|
|
#else /* !(VBOX_XPCOMC_VERSION == 0x00010000U) */
|
2009-05-06 13:51:19 +00:00
|
|
|
data->pFuncs->pfnComInitialize(IVIRTUALBOX_IID_STR, &data->vboxObj,
|
|
|
|
ISESSION_IID_STR, &data->vboxSession);
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2010-03-09 18:22:22 +00:00
|
|
|
# if VBOX_API_VERSION == 2002
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
/* No event queue functionality in 2.2.* as of now */
|
|
|
|
|
2010-03-09 18:22:22 +00:00
|
|
|
# else /* !(VBOX_API_VERSION == 2002) */
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
/* Initial the fWatch needed for Event Callbacks */
|
|
|
|
data->fdWatch = -1;
|
|
|
|
|
|
|
|
data->pFuncs->pfnGetEventQueue(&data->vboxQueue);
|
|
|
|
|
|
|
|
if (data->vboxQueue == NULL) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("nsIEventQueue object is null"));
|
2009-07-24 16:12:16 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif /* !(VBOX_API_VERSION == 2002) */
|
2009-07-24 16:12:16 +00:00
|
|
|
#endif /* !(VBOX_XPCOMC_VERSION == 0x00010000U) */
|
2009-05-06 13:51:19 +00:00
|
|
|
|
|
|
|
if (data->vboxObj == NULL) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("IVirtualBox object is null"));
|
2009-04-17 16:09:07 +00:00
|
|
|
goto cleanup;
|
2009-05-06 13:51:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (data->vboxSession == NULL) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("ISession object is null"));
|
2009-05-06 13:51:19 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-04-02 22:38:57 +00:00
|
|
|
static int vboxExtractVersion(vboxGlobalData *data) {
|
2010-03-30 14:15:13 +00:00
|
|
|
int ret = -1;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUnichar *versionUtf16 = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
|
|
|
|
if (data->version > 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
rc = data->vboxObj->vtbl->GetVersion(data->vboxObj, &versionUtf16);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
char *vboxVersion = NULL;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(versionUtf16, &vboxVersion);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-03-30 14:15:13 +00:00
|
|
|
if (virParseVersionString(vboxVersion, &data->version) >= 0)
|
2009-04-17 16:09:07 +00:00
|
|
|
ret = 0;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(vboxVersion);
|
|
|
|
VBOX_COM_UNALLOC_MEM(versionUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (ret != 0)
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Cound not extract VirtualBox version"));
|
2010-03-30 14:15:13 +00:00
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vboxUninitialize(vboxGlobalData *data) {
|
2009-05-12 15:35:18 +00:00
|
|
|
if (!data)
|
|
|
|
return;
|
|
|
|
|
2009-05-08 10:14:30 +00:00
|
|
|
if (data->pFuncs)
|
|
|
|
data->pFuncs->pfnComUninitialize();
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
virCapabilitiesFree(data->caps);
|
2009-07-24 16:12:16 +00:00
|
|
|
#if VBOX_API_VERSION == 2002
|
|
|
|
/* No domainEventCallbacks in 2.2.* version */
|
|
|
|
#else /* !(VBOX_API_VERSION == 2002) */
|
|
|
|
VIR_FREE(data->domainEventCallbacks);
|
|
|
|
#endif /* !(VBOX_API_VERSION == 2002) */
|
2009-04-17 16:09:07 +00:00
|
|
|
VIR_FREE(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static virDrvOpenStatus vboxOpen(virConnectPtr conn,
|
|
|
|
virConnectAuthPtr auth ATTRIBUTE_UNUSED,
|
|
|
|
int flags ATTRIBUTE_UNUSED) {
|
2009-05-12 15:35:18 +00:00
|
|
|
vboxGlobalData *data = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
uid_t uid = getuid();
|
|
|
|
|
|
|
|
if (conn->uri == NULL) {
|
|
|
|
conn->uri = xmlParseURI(uid ? "vbox:///session" : "vbox:///system");
|
|
|
|
if (conn->uri == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
return VIR_DRV_OPEN_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-12 15:35:18 +00:00
|
|
|
if (conn->uri->scheme == NULL ||
|
|
|
|
STRNEQ (conn->uri->scheme, "vbox"))
|
|
|
|
return VIR_DRV_OPEN_DECLINED;
|
|
|
|
|
|
|
|
/* Leave for remote driver */
|
|
|
|
if (conn->uri->server != NULL)
|
2009-04-17 16:09:07 +00:00
|
|
|
return VIR_DRV_OPEN_DECLINED;
|
|
|
|
|
2009-05-12 15:35:18 +00:00
|
|
|
if (conn->uri->path == NULL || STREQ(conn->uri->path, "")) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2009-05-12 15:35:18 +00:00
|
|
|
_("no VirtualBox driver path specified (try vbox:///session)"));
|
|
|
|
return VIR_DRV_OPEN_ERROR;
|
|
|
|
}
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
if (uid != 0) {
|
2009-05-12 15:35:18 +00:00
|
|
|
if (STRNEQ (conn->uri->path, "/session")) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2009-05-12 15:35:18 +00:00
|
|
|
_("unknown driver path '%s' specified (try vbox:///session)"), conn->uri->path);
|
|
|
|
return VIR_DRV_OPEN_ERROR;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
} else { /* root */
|
|
|
|
if (STRNEQ (conn->uri->path, "/system") &&
|
2009-05-12 15:35:18 +00:00
|
|
|
STRNEQ (conn->uri->path, "/session")) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2009-05-12 15:35:18 +00:00
|
|
|
_("unknown driver path '%s' specified (try vbox:///system)"), conn->uri->path);
|
|
|
|
return VIR_DRV_OPEN_ERROR;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_ALLOC(data) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-05-12 15:35:18 +00:00
|
|
|
return VIR_DRV_OPEN_ERROR;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-05-12 15:35:18 +00:00
|
|
|
if (!(data->caps = vboxCapsInit()) ||
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxInitialize(data) < 0 ||
|
|
|
|
vboxExtractVersion(data) < 0) {
|
2009-05-12 15:35:18 +00:00
|
|
|
vboxUninitialize(data);
|
|
|
|
return VIR_DRV_OPEN_ERROR;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-07-24 16:12:16 +00:00
|
|
|
#if VBOX_API_VERSION == 2002
|
|
|
|
|
|
|
|
/* No domainEventCallbacks in 2.2.* version */
|
|
|
|
|
|
|
|
#else /* !(VBOX_API_VERSION == 2002) */
|
|
|
|
|
|
|
|
if (VIR_ALLOC(data->domainEventCallbacks) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-07-24 16:12:16 +00:00
|
|
|
return VIR_DRV_OPEN_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->conn = conn;
|
|
|
|
g_pVBoxGlobalData = data;
|
|
|
|
|
|
|
|
#endif /* !(VBOX_API_VERSION == 2002) */
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
conn->privateData = data;
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("in vboxOpen");
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
return VIR_DRV_OPEN_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxClose(virConnectPtr conn) {
|
|
|
|
vboxGlobalData *data = conn->privateData;
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("%s: in vboxClose",conn->driver->name);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
vboxUninitialize(data);
|
|
|
|
conn->privateData = NULL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxGetVersion(virConnectPtr conn, unsigned long *version) {
|
|
|
|
vboxGlobalData *data = conn->privateData;
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("%s: in vboxGetVersion",conn->driver->name);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
vboxDriverLock(data);
|
|
|
|
*version = data->version;
|
|
|
|
vboxDriverUnlock(data);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
static int vboxIsSecure(virConnectPtr conn ATTRIBUTE_UNUSED) {
|
|
|
|
/* Driver is using local, non-network based transport */
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxIsEncrypted(virConnectPtr conn ATTRIBUTE_UNUSED) {
|
|
|
|
/* No encryption is needed, or used on the local transport*/
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
static int vboxGetMaxVcpus(virConnectPtr conn, const char *type ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 maxCPUCount = 0;
|
|
|
|
|
|
|
|
/* VirtualBox Supports only hvm and thus the type passed to it
|
|
|
|
* has no meaning, setting it to ATTRIBUTE_UNUSED
|
|
|
|
*/
|
2009-12-04 13:49:45 +00:00
|
|
|
ISystemProperties *systemProperties = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxObj->vtbl->GetSystemProperties(data->vboxObj, &systemProperties);
|
|
|
|
if (systemProperties) {
|
|
|
|
systemProperties->vtbl->GetMaxGuestCPUCount(systemProperties, &maxCPUCount);
|
|
|
|
VBOX_RELEASE(systemProperties);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (maxCPUCount > 0)
|
|
|
|
ret = maxCPUCount;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static char *vboxGetCapabilities(virConnectPtr conn) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, char *, NULL);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
vboxDriverLock(data);
|
|
|
|
ret = virCapabilitiesFormatXML(data->caps);
|
|
|
|
vboxDriverUnlock(data);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxListDomains(virConnectPtr conn, int *ids, int nids) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
int i, j;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get list of Domains, rc=%08x"),(unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
ret = 0;
|
|
|
|
for (i = 0, j = 0; (i < machines.count) && (j < nids); ++i) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (machine) {
|
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
|
|
|
if ( (state >= MachineState_FirstOnline)
|
|
|
|
&& (state <= MachineState_LastOnline) ) {
|
|
|
|
ret++;
|
|
|
|
ids[j++] = i + 1;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxNumOfDomains(virConnectPtr conn) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
int i;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get number of Domains, rc=%08x"), (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
ret = 0;
|
|
|
|
for (i = 0; i < machines.count; ++i) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (machine) {
|
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
|
|
|
if ( (state >= MachineState_FirstOnline)
|
|
|
|
&& (state <= MachineState_LastOnline) )
|
|
|
|
ret++;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainPtr vboxDomainCreateXML(virConnectPtr conn, const char *xml,
|
2010-05-25 17:13:13 +00:00
|
|
|
unsigned int flags) {
|
2009-04-17 16:09:07 +00:00
|
|
|
/* VirtualBox currently doesn't have support for running
|
|
|
|
* virtual machines without actually defining them and thus
|
|
|
|
* for time being just define new machine and start it.
|
|
|
|
*
|
|
|
|
* TODO: After the appropriate API's are added in VirtualBox
|
|
|
|
* change this behaviour to the expected one.
|
|
|
|
*/
|
|
|
|
|
2010-05-25 17:13:13 +00:00
|
|
|
virDomainPtr dom;
|
|
|
|
|
|
|
|
virCheckFlags(0, NULL);
|
|
|
|
|
|
|
|
dom = vboxDomainDefineXML(conn, xml);
|
2009-12-16 12:56:57 +00:00
|
|
|
if (dom == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (vboxDomainCreate(dom) < 0) {
|
|
|
|
vboxDomainUndefine(dom);
|
2010-01-05 16:45:46 +00:00
|
|
|
virUnrefDomain(dom);
|
2009-12-16 12:56:57 +00:00
|
|
|
return NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return dom;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainPtr vboxDomainLookupByID(virConnectPtr conn, int id) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, virDomainPtr, NULL);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-04-21 19:13:23 +00:00
|
|
|
/* Internal vbox IDs start from 0, the public libvirt ID
|
|
|
|
* starts from 1, so refuse id==0, and adjust the rest*/
|
|
|
|
if (id == 0) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2009-04-21 19:13:23 +00:00
|
|
|
_("no domain with matching id %d"), id);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
id = id - 1;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get list of machines, rc=%08x"), (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
if (id < machines.count) {
|
|
|
|
IMachine *machine = machines.items[id];
|
|
|
|
|
|
|
|
if (machine) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
2010-12-16 22:05:48 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (isAccessible) {
|
2010-12-16 22:05:48 +00:00
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-12-04 13:49:45 +00:00
|
|
|
if ( (state >= MachineState_FirstOnline)
|
|
|
|
&& (state <= MachineState_LastOnline) ) {
|
|
|
|
PRUnichar *machineNameUtf16 = NULL;
|
|
|
|
char *machineNameUtf8 = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
machine->vtbl->GetName(machine, &machineNameUtf16);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(machineNameUtf16, &machineNameUtf8);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
machine->vtbl->GetId(machine, &iid.value);
|
|
|
|
vboxIIDToUUID(&iid, uuid);
|
|
|
|
vboxIIDUnalloc(&iid);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
/* get a new domain pointer from virGetDomain, if it fails
|
|
|
|
* then no need to assign the id, else assign the id, cause
|
|
|
|
* it is -1 by default. rest is taken care by virGetDomain
|
|
|
|
* itself, so need not worry.
|
|
|
|
*/
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
ret = virGetDomain(conn, machineNameUtf8, uuid);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (ret)
|
|
|
|
ret->id = id + 1;
|
|
|
|
|
|
|
|
/* Cleanup all the XPCOM allocated stuff here */
|
|
|
|
VBOX_UTF8_FREE(machineNameUtf8);
|
|
|
|
VBOX_UTF16_FREE(machineNameUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
return ret;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainPtr vboxDomainLookupByUUID(virConnectPtr conn, const unsigned char *uuid) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, virDomainPtr, NULL);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-07-24 16:12:16 +00:00
|
|
|
char *machineNameUtf8 = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUnichar *machineNameUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
unsigned char iid_as_uuid[VIR_UUID_BUFLEN];
|
2009-04-17 16:09:07 +00:00
|
|
|
int i, matched = 0;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get list of machines, rc=%08x"), (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < machines.count; ++i) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
continue;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = machine->vtbl->GetId(machine, &iid.value);
|
|
|
|
if (NS_FAILED(rc))
|
2009-12-04 13:49:45 +00:00
|
|
|
continue;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDToUUID(&iid, iid_as_uuid);
|
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (memcmp(uuid, iid_as_uuid, VIR_UUID_BUFLEN) == 0) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 state;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
matched = 1;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetName(machine, &machineNameUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(machineNameUtf16, &machineNameUtf8);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* get a new domain pointer from virGetDomain, if it fails
|
|
|
|
* then no need to assign the id, else assign the id, cause
|
|
|
|
* it is -1 by default. rest is taken care by virGetDomain
|
|
|
|
* itself, so need not worry.
|
|
|
|
*/
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
ret = virGetDomain(conn, machineNameUtf8, iid_as_uuid);
|
2009-12-04 13:49:45 +00:00
|
|
|
if ( ret
|
|
|
|
&& (state >= MachineState_FirstOnline)
|
|
|
|
&& (state <= MachineState_LastOnline) )
|
|
|
|
ret->id = i + 1;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (matched == 1)
|
|
|
|
break;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Do the cleanup and take care you dont leak any memory */
|
|
|
|
VBOX_UTF8_FREE(machineNameUtf8);
|
|
|
|
VBOX_COM_UNALLOC_MEM(machineNameUtf16);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
return ret;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainPtr vboxDomainLookupByName(virConnectPtr conn, const char *name) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, virDomainPtr, NULL);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-07-24 16:12:16 +00:00
|
|
|
char *machineNameUtf8 = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUnichar *machineNameUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
2009-04-17 16:09:07 +00:00
|
|
|
int i, matched = 0;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get list of machines, rc=%08x"), (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < machines.count; ++i) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
continue;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetName(machine, &machineNameUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(machineNameUtf16, &machineNameUtf8);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (STREQ(name, machineNameUtf8)) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 state;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
matched = 1;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
machine->vtbl->GetId(machine, &iid.value);
|
|
|
|
vboxIIDToUUID(&iid, uuid);
|
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* get a new domain pointer from virGetDomain, if it fails
|
|
|
|
* then no need to assign the id, else assign the id, cause
|
|
|
|
* it is -1 by default. rest is taken care by virGetDomain
|
|
|
|
* itself, so need not worry.
|
|
|
|
*/
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
ret = virGetDomain(conn, machineNameUtf8, uuid);
|
2009-12-04 13:49:45 +00:00
|
|
|
if ( ret
|
|
|
|
&& (state >= MachineState_FirstOnline)
|
|
|
|
&& (state <= MachineState_LastOnline) )
|
|
|
|
ret->id = i + 1;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (machineNameUtf8) {
|
|
|
|
VBOX_UTF8_FREE(machineNameUtf8);
|
|
|
|
machineNameUtf8 = NULL;
|
|
|
|
}
|
|
|
|
if (machineNameUtf16) {
|
|
|
|
VBOX_COM_UNALLOC_MEM(machineNameUtf16);
|
|
|
|
machineNameUtf16 = NULL;
|
|
|
|
}
|
|
|
|
if (matched == 1)
|
|
|
|
break;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
return ret;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
|
|
|
static int vboxDomainIsActive(virDomainPtr dom) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
char *machineNameUtf8 = NULL;
|
|
|
|
PRUnichar *machineNameUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
int i, matched = 0;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get list of machines, rc=%08x"), (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
|
|
|
}
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < machines.count; ++i) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
|
|
|
|
if (!machine)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = machine->vtbl->GetId(machine, &iid.value);
|
|
|
|
if (NS_FAILED(rc))
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
continue;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDToUUID(&iid, uuid);
|
|
|
|
vboxIIDUnalloc(&iid);
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (memcmp(dom->uuid, uuid, VIR_UUID_BUFLEN) == 0) {
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 state;
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
matched = 1;
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetName(machine, &machineNameUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(machineNameUtf16, &machineNameUtf8);
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetState(machine, &state);
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if ( (state >= MachineState_FirstOnline)
|
|
|
|
&& (state <= MachineState_LastOnline) )
|
|
|
|
ret = 1;
|
|
|
|
else
|
|
|
|
ret = 0;
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (matched == 1)
|
|
|
|
break;
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Do the cleanup and take care you dont leak any memory */
|
|
|
|
VBOX_UTF8_FREE(machineNameUtf8);
|
|
|
|
VBOX_COM_UNALLOC_MEM(machineNameUtf16);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int vboxDomainIsPersistent(virDomainPtr dom ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
/* All domains are persistent. */
|
Implmentation of new APIs to checking state/persistence of objects
This implements the virConnectIsSecure, virConnectIsEncrypted,
virDomainIsPersistent, virDomainIsActive, virNetworkIsActive,
virNetworkIsPersistent, virStoragePoolIsActive,
virStoragePoolIsPersistent, virInterfaceIsActive APIs in
(nearly) all drivers. Exceptions are:
phyp: missing domainIsActive/Persistent
esx: missing domainIsPersistent
opennebula: missing domainIsActive/Persistent
* src/remote/remote_protocol.x: Define remote wire ABI for newly
added APIs.
* daemon/remote_dispatch*.h: Re-generated from remote_protocol.x
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/network/bridge_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_conf.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/remote/remote_driver.c, src/storage/storage_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xen/xen_driver.h, src/xen/xen_inotify.c,
src/xen/xen_inotify.h: Implement all the new APIs where possible
2009-10-20 14:12:03 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-24 07:43:15 +00:00
|
|
|
static int vboxDomainIsUpdated(virDomainPtr dom ATTRIBUTE_UNUSED) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
static int vboxDomainSuspend(virDomainPtr dom) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
IConsole *console = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching id %d"), dom->id);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (state == MachineState_Running) {
|
2010-12-27 22:35:30 +00:00
|
|
|
/* set state pause */
|
|
|
|
VBOX_SESSION_OPEN_EXISTING(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (console) {
|
|
|
|
console->vtbl->Pause(console);
|
|
|
|
VBOX_RELEASE(console);
|
|
|
|
ret = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("error while suspending the domain"));
|
2009-04-17 16:09:07 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("machine not in running state to suspend it"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainResume(virDomainPtr dom) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
IConsole *console = NULL;
|
|
|
|
PRUint32 state = MachineState_Null;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching id %d"), dom->id);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
|
|
|
|
|
|
|
if (state == MachineState_Paused) {
|
2010-12-27 22:35:30 +00:00
|
|
|
/* resume the machine here */
|
|
|
|
VBOX_SESSION_OPEN_EXISTING(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (console) {
|
|
|
|
console->vtbl->Resume(console);
|
|
|
|
VBOX_RELEASE(console);
|
|
|
|
ret = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("error while resuming the domain"));
|
2009-04-17 16:09:07 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("machine not paused, so can't resume it"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainShutdown(virDomainPtr dom) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
IConsole *console = NULL;
|
|
|
|
PRUint32 state = MachineState_Null;
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching id %d"), dom->id);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (state == MachineState_Paused) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("machine paused, so can't power it down"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
} else if (state == MachineState_PoweredOff) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("machine already powered down"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_OPEN_EXISTING(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (console) {
|
|
|
|
console->vtbl->PowerButton(console);
|
|
|
|
VBOX_RELEASE(console);
|
|
|
|
ret = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainReboot(virDomainPtr dom, unsigned int flags ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
IConsole *console = NULL;
|
|
|
|
PRUint32 state = MachineState_Null;
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching id %d"), dom->id);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (state == MachineState_Running) {
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_OPEN_EXISTING(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (console) {
|
|
|
|
console->vtbl->Reset(console);
|
|
|
|
VBOX_RELEASE(console);
|
|
|
|
ret = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("machine not running, so can't reboot it"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainDestroy(virDomainPtr dom) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
IConsole *console = NULL;
|
|
|
|
PRUint32 state = MachineState_Null;
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching id %d"), dom->id);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (state == MachineState_PoweredOff) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("machine already powered down"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_OPEN_EXISTING(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (console) {
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
#if VBOX_API_VERSION == 2002
|
2009-12-04 13:49:45 +00:00
|
|
|
console->vtbl->PowerDown(console);
|
2009-07-24 16:12:16 +00:00
|
|
|
#else
|
2010-03-31 08:39:48 +00:00
|
|
|
IProgress *progress = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
console->vtbl->PowerDown(console, &progress);
|
|
|
|
if (progress) {
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
VBOX_RELEASE(progress);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif
|
|
|
|
VBOX_RELEASE(console);
|
2010-07-30 17:50:12 +00:00
|
|
|
dom->id = -1;
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-02-04 18:19:08 +00:00
|
|
|
static char *vboxDomainGetOSType(virDomainPtr dom ATTRIBUTE_UNUSED) {
|
2009-04-17 16:09:07 +00:00
|
|
|
/* Returning "hvm" always as suggested on list, cause
|
|
|
|
* this functions seems to be badly named and it
|
|
|
|
* is supposed to pass the ABI name and not the domain
|
|
|
|
* operating system driver as I had imagined ;)
|
|
|
|
*/
|
2009-11-08 21:08:54 +00:00
|
|
|
char *osType = strdup("hvm");
|
|
|
|
|
|
|
|
if (osType == NULL)
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-11-08 21:08:54 +00:00
|
|
|
|
|
|
|
return osType;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainSetMemory(virDomainPtr dom, unsigned long memory) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state = MachineState_Null;
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching id %d"), dom->id);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (state != MachineState_PoweredOff) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("memory size can't be changed unless domain is powered down"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
rc = data->vboxSession->vtbl->GetMachine(data->vboxSession, &machine);
|
|
|
|
if (NS_SUCCEEDED(rc) && machine) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2011-01-28 21:03:24 +00:00
|
|
|
rc = machine->vtbl->SetMemorySize(machine,
|
|
|
|
VIR_DIV_UP(memory, 1024));
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
machine->vtbl->SaveSettings(machine);
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not set the memory size of the "
|
|
|
|
"domain to: %lu Kb, rc=%08x"),
|
2009-12-04 13:49:45 +00:00
|
|
|
memory, (unsigned)rc);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainGetInfo(virDomainPtr dom, virDomainInfoPtr info) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
char *machineName = NULL;
|
|
|
|
PRUnichar *machineNameUtf16 = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
|
|
|
int i = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get list of machines, rc=%08x"), (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
info->nrVirtCpu = 0;
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < machines.count; ++i) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
continue;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetName(machine, &machineNameUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(machineNameUtf16, &machineName);
|
|
|
|
|
|
|
|
if (STREQ(dom->name, machineName)) {
|
|
|
|
/* Get the Machine State (also match it with
|
|
|
|
* virDomainState). Get the Machine memory and
|
2010-10-12 14:43:39 +00:00
|
|
|
* for time being set max_balloon and cur_balloon to same
|
2009-12-04 13:49:45 +00:00
|
|
|
* Also since there is no direct way of checking
|
|
|
|
* the cputime required (one condition being the
|
|
|
|
* VM is remote), return zero for cputime. Get the
|
|
|
|
* number of CPU.
|
|
|
|
*/
|
|
|
|
PRUint32 CPUCount = 0;
|
|
|
|
PRUint32 memorySize = 0;
|
|
|
|
PRUint32 state = MachineState_Null;
|
|
|
|
PRUint32 maxMemorySize = 4 * 1024;
|
|
|
|
ISystemProperties *systemProperties = NULL;
|
|
|
|
|
|
|
|
data->vboxObj->vtbl->GetSystemProperties(data->vboxObj, &systemProperties);
|
|
|
|
if (systemProperties) {
|
|
|
|
systemProperties->vtbl->GetMaxGuestRAM(systemProperties, &maxMemorySize);
|
|
|
|
VBOX_RELEASE(systemProperties);
|
|
|
|
systemProperties = NULL;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetCPUCount(machine, &CPUCount);
|
|
|
|
machine->vtbl->GetMemorySize(machine, &memorySize);
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
|
|
|
|
|
|
|
info->cpuTime = 0;
|
|
|
|
info->nrVirtCpu = CPUCount;
|
2010-10-12 19:24:11 +00:00
|
|
|
info->memory = memorySize * 1024;
|
|
|
|
info->maxMem = maxMemorySize * 1024;
|
2009-12-04 13:49:45 +00:00
|
|
|
switch(state) {
|
|
|
|
case MachineState_Running:
|
|
|
|
info->state = VIR_DOMAIN_RUNNING;
|
|
|
|
break;
|
|
|
|
case MachineState_Stuck:
|
|
|
|
info->state = VIR_DOMAIN_BLOCKED;
|
|
|
|
break;
|
|
|
|
case MachineState_Paused:
|
|
|
|
info->state = VIR_DOMAIN_PAUSED;
|
|
|
|
break;
|
|
|
|
case MachineState_Stopping:
|
|
|
|
info->state = VIR_DOMAIN_SHUTDOWN;
|
|
|
|
break;
|
|
|
|
case MachineState_PoweredOff:
|
|
|
|
info->state = VIR_DOMAIN_SHUTOFF;
|
|
|
|
break;
|
|
|
|
case MachineState_Aborted:
|
|
|
|
info->state = VIR_DOMAIN_CRASHED;
|
|
|
|
break;
|
|
|
|
case MachineState_Null:
|
|
|
|
default:
|
|
|
|
info->state = VIR_DOMAIN_NOSTATE;
|
|
|
|
break;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (machineName)
|
|
|
|
VBOX_UTF8_FREE(machineName);
|
|
|
|
if (machineNameUtf16)
|
|
|
|
VBOX_COM_UNALLOC_MEM(machineNameUtf16);
|
|
|
|
if (info->nrVirtCpu)
|
|
|
|
break;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
cleanup:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainSave(virDomainPtr dom, const char *path ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IConsole *console = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2010-12-27 22:35:30 +00:00
|
|
|
IMachine *machine = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
/* VirtualBox currently doesn't support saving to a file
|
|
|
|
* at a location other then the machine folder and thus
|
|
|
|
* setting path to ATTRIBUTE_UNUSED for now, will change
|
|
|
|
* this behaviour once get the VirtualBox API in right
|
|
|
|
* shape to do this
|
|
|
|
*/
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Open a Session for the machine */
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION >= 4000
|
|
|
|
/* Get machine for the call to VBOX_SESSION_OPEN_EXISTING */
|
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
|
|
|
_("no domain with matching uuid"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
rc = VBOX_SESSION_OPEN_EXISTING(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
rc = data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (NS_SUCCEEDED(rc) && console) {
|
|
|
|
IProgress *progress = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
console->vtbl->SaveState(console, &progress);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (progress) {
|
2009-07-24 16:12:16 +00:00
|
|
|
#if VBOX_API_VERSION == 2002
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult resultCode;
|
2009-07-24 16:12:16 +00:00
|
|
|
#else
|
2009-12-04 13:49:45 +00:00
|
|
|
PRInt32 resultCode;
|
2009-07-24 16:12:16 +00:00
|
|
|
#endif
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
progress->vtbl->GetResultCode(progress, &resultCode);
|
|
|
|
if (NS_SUCCEEDED(resultCode)) {
|
|
|
|
ret = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(progress);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(console);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("UUID of machine being saved:", iid.value);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
vcpu: make old API trivially wrap to new API
Note - this wrapping is completely mechanical; the old API will
function identically, since the new API validates that the exact
same flags are provided by the old API. On a per-driver basis,
it may make sense to have the old API pass a different set of flags,
but that should be done in the per-driver patch that implements
the full range of flag support in the new API.
* src/esx/esx_driver.c (esxDomainSetVcpus, escDomainGetMaxVpcus):
Move guts...
(esxDomainSetVcpusFlags, esxDomainGetVcpusFlags): ...to new
functions.
(esxDriver): Trivially support the new API.
* src/openvz/openvz_driver.c (openvzDomainSetVcpus)
(openvzDomainSetVcpusFlags, openvzDomainGetMaxVcpus)
(openvzDomainGetVcpusFlags, openvzDriver): Likewise.
* src/phyp/phyp_driver.c (phypDomainSetCPU)
(phypDomainSetVcpusFlags, phypGetLparCPUMAX)
(phypDomainGetVcpusFlags, phypDriver): Likewise.
* src/qemu/qemu_driver.c (qemudDomainSetVcpus)
(qemudDomainSetVcpusFlags, qemudDomainGetMaxVcpus)
(qemudDomainGetVcpusFlags, qemuDriver): Likewise.
* src/test/test_driver.c (testSetVcpus, testDomainSetVcpusFlags)
(testDomainGetMaxVcpus, testDomainGetVcpusFlags, testDriver):
Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainSetVcpus)
(vboxDomainSetVcpusFlags, virDomainGetMaxVcpus)
(virDomainGetVcpusFlags, virDriver): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainSetVcpus)
(xenUnifiedDomainSetVcpusFlags, xenUnifiedDomainGetMaxVcpus)
(xenUnifiedDomainGetVcpusFlags, xenUnifiedDriver): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainSetVcpus)
(xenapiDomainSetVcpusFlags, xenapiDomainGetMaxVcpus)
(xenapiDomainGetVcpusFlags, xenapiDriver): Likewise.
(xenapiError): New helper macro.
2010-09-27 22:37:53 +00:00
|
|
|
static int
|
|
|
|
vboxDomainSetVcpusFlags(virDomainPtr dom, unsigned int nvcpus,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-09-02 14:08:14 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-09-02 14:08:14 +00:00
|
|
|
PRUint32 CPUCount = nvcpus;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-09-02 14:08:14 +00:00
|
|
|
|
vcpu: make old API trivially wrap to new API
Note - this wrapping is completely mechanical; the old API will
function identically, since the new API validates that the exact
same flags are provided by the old API. On a per-driver basis,
it may make sense to have the old API pass a different set of flags,
but that should be done in the per-driver patch that implements
the full range of flag support in the new API.
* src/esx/esx_driver.c (esxDomainSetVcpus, escDomainGetMaxVpcus):
Move guts...
(esxDomainSetVcpusFlags, esxDomainGetVcpusFlags): ...to new
functions.
(esxDriver): Trivially support the new API.
* src/openvz/openvz_driver.c (openvzDomainSetVcpus)
(openvzDomainSetVcpusFlags, openvzDomainGetMaxVcpus)
(openvzDomainGetVcpusFlags, openvzDriver): Likewise.
* src/phyp/phyp_driver.c (phypDomainSetCPU)
(phypDomainSetVcpusFlags, phypGetLparCPUMAX)
(phypDomainGetVcpusFlags, phypDriver): Likewise.
* src/qemu/qemu_driver.c (qemudDomainSetVcpus)
(qemudDomainSetVcpusFlags, qemudDomainGetMaxVcpus)
(qemudDomainGetVcpusFlags, qemuDriver): Likewise.
* src/test/test_driver.c (testSetVcpus, testDomainSetVcpusFlags)
(testDomainGetMaxVcpus, testDomainGetVcpusFlags, testDriver):
Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainSetVcpus)
(vboxDomainSetVcpusFlags, virDomainGetMaxVcpus)
(virDomainGetVcpusFlags, virDriver): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainSetVcpus)
(xenUnifiedDomainSetVcpusFlags, xenUnifiedDomainGetMaxVcpus)
(xenUnifiedDomainGetVcpusFlags, xenUnifiedDriver): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainSetVcpus)
(xenapiDomainSetVcpusFlags, xenapiDomainGetMaxVcpus)
(xenapiDomainGetVcpusFlags, xenapiDriver): Likewise.
(xenapiError): New helper macro.
2010-09-27 22:37:53 +00:00
|
|
|
if (flags != VIR_DOMAIN_VCPU_LIVE) {
|
|
|
|
vboxError(VIR_ERR_INVALID_ARG, _("unsupported flags: (0x%x)"), flags);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION >= 4000
|
|
|
|
/* Get machine for the call to VBOX_SESSION_OPEN */
|
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
|
|
|
_("no domain with matching uuid"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
rc = VBOX_SESSION_OPEN(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
data->vboxSession->vtbl->GetMachine(data->vboxSession, &machine);
|
|
|
|
if (machine) {
|
|
|
|
rc = machine->vtbl->SetCPUCount(machine, CPUCount);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
machine->vtbl->SaveSettings(machine);
|
|
|
|
ret = 0;
|
2009-09-02 14:08:14 +00:00
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not set the number of cpus of the domain "
|
|
|
|
"to: %u, rc=%08x"),
|
2009-12-04 13:49:45 +00:00
|
|
|
CPUCount, (unsigned)rc);
|
2009-09-02 14:08:14 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2009-09-02 14:08:14 +00:00
|
|
|
} else {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching id %d"), dom->id);
|
2009-09-02 14:08:14 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("can't open session to the domain with id %d"), dom->id);
|
2009-09-02 14:08:14 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-09-02 14:08:14 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-09-02 14:08:14 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
vcpu: make old API trivially wrap to new API
Note - this wrapping is completely mechanical; the old API will
function identically, since the new API validates that the exact
same flags are provided by the old API. On a per-driver basis,
it may make sense to have the old API pass a different set of flags,
but that should be done in the per-driver patch that implements
the full range of flag support in the new API.
* src/esx/esx_driver.c (esxDomainSetVcpus, escDomainGetMaxVpcus):
Move guts...
(esxDomainSetVcpusFlags, esxDomainGetVcpusFlags): ...to new
functions.
(esxDriver): Trivially support the new API.
* src/openvz/openvz_driver.c (openvzDomainSetVcpus)
(openvzDomainSetVcpusFlags, openvzDomainGetMaxVcpus)
(openvzDomainGetVcpusFlags, openvzDriver): Likewise.
* src/phyp/phyp_driver.c (phypDomainSetCPU)
(phypDomainSetVcpusFlags, phypGetLparCPUMAX)
(phypDomainGetVcpusFlags, phypDriver): Likewise.
* src/qemu/qemu_driver.c (qemudDomainSetVcpus)
(qemudDomainSetVcpusFlags, qemudDomainGetMaxVcpus)
(qemudDomainGetVcpusFlags, qemuDriver): Likewise.
* src/test/test_driver.c (testSetVcpus, testDomainSetVcpusFlags)
(testDomainGetMaxVcpus, testDomainGetVcpusFlags, testDriver):
Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainSetVcpus)
(vboxDomainSetVcpusFlags, virDomainGetMaxVcpus)
(virDomainGetVcpusFlags, virDriver): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainSetVcpus)
(xenUnifiedDomainSetVcpusFlags, xenUnifiedDomainGetMaxVcpus)
(xenUnifiedDomainGetVcpusFlags, xenUnifiedDriver): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainSetVcpus)
(xenapiDomainSetVcpusFlags, xenapiDomainGetMaxVcpus)
(xenapiDomainGetVcpusFlags, xenapiDriver): Likewise.
(xenapiError): New helper macro.
2010-09-27 22:37:53 +00:00
|
|
|
static int
|
|
|
|
vboxDomainSetVcpus(virDomainPtr dom, unsigned int nvcpus)
|
|
|
|
{
|
|
|
|
return vboxDomainSetVcpusFlags(dom, nvcpus, VIR_DOMAIN_VCPU_LIVE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxDomainGetVcpusFlags(virDomainPtr dom, unsigned int flags)
|
|
|
|
{
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
|
|
|
ISystemProperties *systemProperties = NULL;
|
2009-09-02 14:08:14 +00:00
|
|
|
PRUint32 maxCPUCount = 0;
|
|
|
|
|
vcpu: make old API trivially wrap to new API
Note - this wrapping is completely mechanical; the old API will
function identically, since the new API validates that the exact
same flags are provided by the old API. On a per-driver basis,
it may make sense to have the old API pass a different set of flags,
but that should be done in the per-driver patch that implements
the full range of flag support in the new API.
* src/esx/esx_driver.c (esxDomainSetVcpus, escDomainGetMaxVpcus):
Move guts...
(esxDomainSetVcpusFlags, esxDomainGetVcpusFlags): ...to new
functions.
(esxDriver): Trivially support the new API.
* src/openvz/openvz_driver.c (openvzDomainSetVcpus)
(openvzDomainSetVcpusFlags, openvzDomainGetMaxVcpus)
(openvzDomainGetVcpusFlags, openvzDriver): Likewise.
* src/phyp/phyp_driver.c (phypDomainSetCPU)
(phypDomainSetVcpusFlags, phypGetLparCPUMAX)
(phypDomainGetVcpusFlags, phypDriver): Likewise.
* src/qemu/qemu_driver.c (qemudDomainSetVcpus)
(qemudDomainSetVcpusFlags, qemudDomainGetMaxVcpus)
(qemudDomainGetVcpusFlags, qemuDriver): Likewise.
* src/test/test_driver.c (testSetVcpus, testDomainSetVcpusFlags)
(testDomainGetMaxVcpus, testDomainGetVcpusFlags, testDriver):
Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainSetVcpus)
(vboxDomainSetVcpusFlags, virDomainGetMaxVcpus)
(virDomainGetVcpusFlags, virDriver): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainSetVcpus)
(xenUnifiedDomainSetVcpusFlags, xenUnifiedDomainGetMaxVcpus)
(xenUnifiedDomainGetVcpusFlags, xenUnifiedDriver): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainSetVcpus)
(xenapiDomainSetVcpusFlags, xenapiDomainGetMaxVcpus)
(xenapiDomainGetVcpusFlags, xenapiDriver): Likewise.
(xenapiError): New helper macro.
2010-09-27 22:37:53 +00:00
|
|
|
if (flags != (VIR_DOMAIN_VCPU_LIVE | VIR_DOMAIN_VCPU_MAXIMUM)) {
|
|
|
|
vboxError(VIR_ERR_INVALID_ARG, _("unsupported flags: (0x%x)"), flags);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-09-02 14:08:14 +00:00
|
|
|
/* Currently every domain supports the same number of max cpus
|
|
|
|
* as that supported by vbox and thus take it directly from
|
|
|
|
* the systemproperties.
|
|
|
|
*/
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxObj->vtbl->GetSystemProperties(data->vboxObj, &systemProperties);
|
|
|
|
if (systemProperties) {
|
|
|
|
systemProperties->vtbl->GetMaxGuestCPUCount(systemProperties, &maxCPUCount);
|
|
|
|
VBOX_RELEASE(systemProperties);
|
2009-09-02 14:08:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (maxCPUCount > 0)
|
|
|
|
ret = maxCPUCount;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
vcpu: make old API trivially wrap to new API
Note - this wrapping is completely mechanical; the old API will
function identically, since the new API validates that the exact
same flags are provided by the old API. On a per-driver basis,
it may make sense to have the old API pass a different set of flags,
but that should be done in the per-driver patch that implements
the full range of flag support in the new API.
* src/esx/esx_driver.c (esxDomainSetVcpus, escDomainGetMaxVpcus):
Move guts...
(esxDomainSetVcpusFlags, esxDomainGetVcpusFlags): ...to new
functions.
(esxDriver): Trivially support the new API.
* src/openvz/openvz_driver.c (openvzDomainSetVcpus)
(openvzDomainSetVcpusFlags, openvzDomainGetMaxVcpus)
(openvzDomainGetVcpusFlags, openvzDriver): Likewise.
* src/phyp/phyp_driver.c (phypDomainSetCPU)
(phypDomainSetVcpusFlags, phypGetLparCPUMAX)
(phypDomainGetVcpusFlags, phypDriver): Likewise.
* src/qemu/qemu_driver.c (qemudDomainSetVcpus)
(qemudDomainSetVcpusFlags, qemudDomainGetMaxVcpus)
(qemudDomainGetVcpusFlags, qemuDriver): Likewise.
* src/test/test_driver.c (testSetVcpus, testDomainSetVcpusFlags)
(testDomainGetMaxVcpus, testDomainGetVcpusFlags, testDriver):
Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainSetVcpus)
(vboxDomainSetVcpusFlags, virDomainGetMaxVcpus)
(virDomainGetVcpusFlags, virDriver): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainSetVcpus)
(xenUnifiedDomainSetVcpusFlags, xenUnifiedDomainGetMaxVcpus)
(xenUnifiedDomainGetVcpusFlags, xenUnifiedDriver): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainSetVcpus)
(xenapiDomainSetVcpusFlags, xenapiDomainGetMaxVcpus)
(xenapiDomainGetVcpusFlags, xenapiDriver): Likewise.
(xenapiError): New helper macro.
2010-09-27 22:37:53 +00:00
|
|
|
static int
|
|
|
|
vboxDomainGetMaxVcpus(virDomainPtr dom)
|
|
|
|
{
|
|
|
|
return vboxDomainGetVcpusFlags(dom, (VIR_DOMAIN_VCPU_LIVE |
|
|
|
|
VIR_DOMAIN_VCPU_MAXIMUM));
|
|
|
|
}
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
static char *vboxDomainDumpXML(virDomainPtr dom, int flags) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, char *, NULL);
|
2009-04-17 16:09:07 +00:00
|
|
|
virDomainDefPtr def = NULL;
|
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
int gotAllABoutDef = -1;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2010-06-10 20:11:39 +00:00
|
|
|
char *tmp;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
if (VIR_ALLOC(def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool accessible = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &accessible);
|
|
|
|
if (accessible) {
|
|
|
|
int i = 0;
|
|
|
|
struct utsname utsname;
|
|
|
|
PRBool PAEEnabled = PR_FALSE;
|
|
|
|
PRBool ACPIEnabled = PR_FALSE;
|
|
|
|
PRBool IOAPICEnabled = PR_FALSE;
|
2010-12-27 22:35:30 +00:00
|
|
|
PRBool VRDxEnabled = PR_FALSE;
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 CPUCount = 0;
|
|
|
|
PRUint32 memorySize = 0;
|
|
|
|
PRUint32 netAdpCnt = 0;
|
|
|
|
PRUint32 netAdpIncCnt = 0;
|
|
|
|
PRUint32 maxMemorySize = 4 * 1024;
|
|
|
|
PRUint32 USBFilterCount = 0;
|
|
|
|
PRUint32 maxBootPosition = 0;
|
|
|
|
PRUint32 serialPortCount = 0;
|
|
|
|
PRUint32 serialPortIncCount = 0;
|
|
|
|
PRUint32 parallelPortCount = 0;
|
|
|
|
PRUint32 parallelPortIncCount = 0;
|
|
|
|
IBIOSSettings *bios = NULL;
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
PRInt32 hddNum = 0;
|
|
|
|
IDVDDrive *dvdDrive = NULL;
|
|
|
|
IHardDisk *hardDiskPM = NULL;
|
|
|
|
IHardDisk *hardDiskPS = NULL;
|
|
|
|
IHardDisk *hardDiskSS = NULL;
|
|
|
|
const char *hddBus = "IDE";
|
|
|
|
PRUnichar *hddBusUtf16 = NULL;
|
|
|
|
IFloppyDrive *floppyDrive = NULL;
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray mediumAttachments = VBOX_ARRAY_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
|
|
|
IVRDPServer *VRDxServer = NULL;
|
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
IVRDEServer *VRDxServer = NULL;
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
IAudioAdapter *audioAdapter = NULL;
|
|
|
|
IUSBController *USBController = NULL;
|
|
|
|
ISystemProperties *systemProperties = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->virtType = VIR_DOMAIN_VIRT_VBOX;
|
|
|
|
def->id = dom->id;
|
|
|
|
memcpy(def->uuid, dom->uuid, VIR_UUID_BUFLEN);
|
|
|
|
def->name = strdup(dom->name);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetMemorySize(machine, &memorySize);
|
2010-10-12 14:43:39 +00:00
|
|
|
def->mem.cur_balloon = memorySize * 1024;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxObj->vtbl->GetSystemProperties(data->vboxObj, &systemProperties);
|
|
|
|
if (systemProperties) {
|
|
|
|
systemProperties->vtbl->GetMaxGuestRAM(systemProperties, &maxMemorySize);
|
|
|
|
systemProperties->vtbl->GetMaxBootPosition(systemProperties, &maxBootPosition);
|
|
|
|
systemProperties->vtbl->GetNetworkAdapterCount(systemProperties, &netAdpCnt);
|
|
|
|
systemProperties->vtbl->GetSerialPortCount(systemProperties, &serialPortCount);
|
|
|
|
systemProperties->vtbl->GetParallelPortCount(systemProperties, ¶llelPortCount);
|
|
|
|
VBOX_RELEASE(systemProperties);
|
|
|
|
systemProperties = NULL;
|
|
|
|
}
|
|
|
|
/* Currently setting memory and maxMemory as same, cause
|
|
|
|
* the notation here seems to be inconsistent while
|
|
|
|
* reading and while dumping xml
|
|
|
|
*/
|
2010-10-12 14:43:39 +00:00
|
|
|
/* def->mem.max_balloon = maxMemorySize * 1024; */
|
|
|
|
def->mem.max_balloon = memorySize * 1024;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
machine->vtbl->GetCPUCount(machine, &CPUCount);
|
2010-09-29 16:20:07 +00:00
|
|
|
def->maxvcpus = def->vcpus = CPUCount;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
/* Skip cpumasklen, cpumask, onReboot, onPoweroff, onCrash */
|
|
|
|
|
|
|
|
def->os.type = strdup("hvm");
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
uname(&utsname);
|
|
|
|
def->os.arch = strdup(utsname.machine);
|
|
|
|
|
|
|
|
def->os.nBootDevs = 0;
|
|
|
|
for (i = 0; (i < VIR_DOMAIN_BOOT_LAST) && (i < maxBootPosition); i++) {
|
|
|
|
PRUint32 device = DeviceType_Null;
|
|
|
|
|
|
|
|
machine->vtbl->GetBootOrder(machine, i+1, &device);
|
|
|
|
|
|
|
|
if (device == DeviceType_Floppy) {
|
|
|
|
def->os.bootDevs[i] = VIR_DOMAIN_BOOT_FLOPPY;
|
|
|
|
def->os.nBootDevs++;
|
|
|
|
} else if (device == DeviceType_DVD) {
|
|
|
|
def->os.bootDevs[i] = VIR_DOMAIN_BOOT_CDROM;
|
|
|
|
def->os.nBootDevs++;
|
|
|
|
} else if (device == DeviceType_HardDisk) {
|
|
|
|
def->os.bootDevs[i] = VIR_DOMAIN_BOOT_DISK;
|
|
|
|
def->os.nBootDevs++;
|
|
|
|
} else if (device == DeviceType_Network) {
|
|
|
|
def->os.bootDevs[i] = VIR_DOMAIN_BOOT_NET;
|
|
|
|
def->os.nBootDevs++;
|
|
|
|
} else if (device == DeviceType_USB) {
|
|
|
|
/* Not supported by libvirt yet */
|
|
|
|
} else if (device == DeviceType_SharedFolder) {
|
|
|
|
/* Not supported by libvirt yet */
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->features = 0;
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
machine->vtbl->GetPAEEnabled(machine, &PAEEnabled);
|
2010-05-26 11:54:16 +00:00
|
|
|
#elif VBOX_API_VERSION == 3001
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetCpuProperty(machine, CpuPropertyType_PAE, &PAEEnabled);
|
2010-05-26 11:54:16 +00:00
|
|
|
#elif VBOX_API_VERSION >= 3002
|
|
|
|
machine->vtbl->GetCPUProperty(machine, CPUPropertyType_PAE, &PAEEnabled);
|
|
|
|
#endif
|
2009-12-04 13:49:45 +00:00
|
|
|
if (PAEEnabled) {
|
|
|
|
def->features = def->features | (1 << VIR_DOMAIN_FEATURE_PAE);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetBIOSSettings(machine, &bios);
|
|
|
|
if (bios) {
|
|
|
|
bios->vtbl->GetACPIEnabled(bios, &ACPIEnabled);
|
|
|
|
if (ACPIEnabled) {
|
|
|
|
def->features = def->features | (1 << VIR_DOMAIN_FEATURE_ACPI);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
bios->vtbl->GetIOAPICEnabled(bios, &IOAPICEnabled);
|
|
|
|
if (IOAPICEnabled) {
|
|
|
|
def->features = def->features | (1 << VIR_DOMAIN_FEATURE_APIC);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(bios);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Currently VirtualBox always uses locatime
|
|
|
|
* so locatime is always true here */
|
2010-02-02 17:22:03 +00:00
|
|
|
def->clock.offset = VIR_DOMAIN_CLOCK_OFFSET_LOCALTIME;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
/* dump video options vram/2d/3d/directx/etc. */
|
|
|
|
{
|
|
|
|
/* Currently supports only one graphics card */
|
|
|
|
def->nvideos = 1;
|
|
|
|
if (VIR_ALLOC_N(def->videos, def->nvideos) >= 0) {
|
|
|
|
if (VIR_ALLOC(def->videos[0]) >= 0) {
|
|
|
|
/* the default is: vram is 8MB, One monitor, 3dAccel Off */
|
2010-12-30 14:18:27 +00:00
|
|
|
PRUint32 VRAMSize = 8;
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 monitorCount = 1;
|
|
|
|
PRBool accelerate3DEnabled = PR_FALSE;
|
|
|
|
PRBool accelerate2DEnabled = PR_FALSE;
|
|
|
|
|
|
|
|
machine->vtbl->GetVRAMSize(machine, &VRAMSize);
|
|
|
|
machine->vtbl->GetMonitorCount(machine, &monitorCount);
|
|
|
|
machine->vtbl->GetAccelerate3DEnabled(machine, &accelerate3DEnabled);
|
|
|
|
#if VBOX_API_VERSION >= 3001
|
|
|
|
machine->vtbl->GetAccelerate2DVideoEnabled(machine, &accelerate2DEnabled);
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
|
|
|
|
|
|
|
def->videos[0]->type = VIR_DOMAIN_VIDEO_TYPE_VBOX;
|
2010-12-30 14:18:27 +00:00
|
|
|
def->videos[0]->vram = VRAMSize * 1024;
|
2009-12-04 13:49:45 +00:00
|
|
|
def->videos[0]->heads = monitorCount;
|
|
|
|
if (VIR_ALLOC(def->videos[0]->accel) >= 0) {
|
|
|
|
def->videos[0]->accel->support3d = accelerate3DEnabled;
|
|
|
|
def->videos[0]->accel->support2d = accelerate2DEnabled;
|
2009-09-07 15:44:12 +00:00
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-09-07 15:44:12 +00:00
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-09-03 08:36:44 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* dump display options vrdp/gui/sdl */
|
|
|
|
{
|
|
|
|
int vrdpPresent = 0;
|
|
|
|
int sdlPresent = 0;
|
|
|
|
int guiPresent = 0;
|
|
|
|
int totalPresent = 0;
|
|
|
|
char *guiDisplay = NULL;
|
|
|
|
char *sdlDisplay = NULL;
|
|
|
|
PRUnichar *keyTypeUtf16 = NULL;
|
|
|
|
PRUnichar *valueTypeUtf16 = NULL;
|
|
|
|
char *valueTypeUtf8 = NULL;
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->ngraphics = 0;
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Type", &keyTypeUtf16);
|
|
|
|
machine->vtbl->GetExtraData(machine, keyTypeUtf16, &valueTypeUtf16);
|
|
|
|
VBOX_UTF16_FREE(keyTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (valueTypeUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(valueTypeUtf16, &valueTypeUtf8);
|
|
|
|
VBOX_UTF16_FREE(valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if ( STREQ(valueTypeUtf8, "sdl") || STREQ(valueTypeUtf8, "gui") ) {
|
|
|
|
PRUnichar *keyDislpayUtf16 = NULL;
|
|
|
|
PRUnichar *valueDisplayUtf16 = NULL;
|
|
|
|
char *valueDisplayUtf8 = NULL;
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Display", &keyDislpayUtf16);
|
|
|
|
machine->vtbl->GetExtraData(machine, keyDislpayUtf16, &valueDisplayUtf16);
|
|
|
|
VBOX_UTF16_FREE(keyDislpayUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (valueDisplayUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(valueDisplayUtf16, &valueDisplayUtf8);
|
|
|
|
VBOX_UTF16_FREE(valueDisplayUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (strlen(valueDisplayUtf8) <= 0) {
|
|
|
|
VBOX_UTF8_FREE(valueDisplayUtf8);
|
|
|
|
valueDisplayUtf8 = NULL;
|
2009-05-15 13:45:40 +00:00
|
|
|
}
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (STREQ(valueTypeUtf8, "sdl")) {
|
|
|
|
sdlPresent = 1;
|
|
|
|
if (valueDisplayUtf8)
|
|
|
|
sdlDisplay = strdup(valueDisplayUtf8);
|
|
|
|
if (sdlDisplay == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
/* just don't go to cleanup yet as it is ok to have
|
|
|
|
* sdlDisplay as NULL and we check it below if it
|
|
|
|
* exist and then only use it there
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
totalPresent++;
|
2009-05-15 13:45:40 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (STREQ(valueTypeUtf8, "gui")) {
|
|
|
|
guiPresent = 1;
|
|
|
|
if (valueDisplayUtf8)
|
|
|
|
guiDisplay = strdup(valueDisplayUtf8);
|
|
|
|
if (guiDisplay == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-05-15 13:45:40 +00:00
|
|
|
/* just don't go to cleanup yet as it is ok to have
|
2009-12-04 13:49:45 +00:00
|
|
|
* guiDisplay as NULL and we check it below if it
|
|
|
|
* exist and then only use it there
|
2009-05-15 13:45:40 +00:00
|
|
|
*/
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
totalPresent++;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
if (valueDisplayUtf8)
|
|
|
|
VBOX_UTF8_FREE(valueDisplayUtf8);
|
2009-05-15 13:45:40 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (STREQ(valueTypeUtf8, "vrdp"))
|
|
|
|
vrdpPresent = 1;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(valueTypeUtf8);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if ((totalPresent > 0) && (VIR_ALLOC_N(def->graphics, totalPresent) >= 0)) {
|
|
|
|
if ((guiPresent) && (VIR_ALLOC(def->graphics[def->ngraphics]) >= 0)) {
|
|
|
|
def->graphics[def->ngraphics]->type = VIR_DOMAIN_GRAPHICS_TYPE_DESKTOP;
|
|
|
|
if (guiDisplay)
|
|
|
|
def->graphics[def->ngraphics]->data.desktop.display = guiDisplay;
|
|
|
|
def->ngraphics++;
|
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if ((sdlPresent) && (VIR_ALLOC(def->graphics[def->ngraphics]) >= 0)) {
|
|
|
|
def->graphics[def->ngraphics]->type = VIR_DOMAIN_GRAPHICS_TYPE_SDL;
|
|
|
|
if (sdlDisplay)
|
|
|
|
def->graphics[def->ngraphics]->data.sdl.display = sdlDisplay;
|
|
|
|
def->ngraphics++;
|
|
|
|
}
|
|
|
|
} else if ((vrdpPresent != 1) && (totalPresent == 0) && (VIR_ALLOC_N(def->graphics, 1) >= 0)) {
|
|
|
|
if (VIR_ALLOC(def->graphics[def->ngraphics]) >= 0) {
|
|
|
|
def->graphics[def->ngraphics]->type = VIR_DOMAIN_GRAPHICS_TYPE_DESKTOP;
|
2010-06-10 20:11:39 +00:00
|
|
|
tmp = getenv("DISPLAY");
|
|
|
|
if (tmp != NULL) {
|
|
|
|
def->graphics[def->ngraphics]->data.desktop.display = strdup(tmp);
|
|
|
|
if (def->graphics[def->ngraphics]->data.desktop.display == NULL) {
|
|
|
|
virReportOOMError();
|
|
|
|
/* just don't go to cleanup yet as it is ok to have
|
|
|
|
* display as NULL
|
|
|
|
*/
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
totalPresent++;
|
|
|
|
def->ngraphics++;
|
|
|
|
}
|
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
|
|
|
machine->vtbl->GetVRDPServer(machine, &VRDxServer);
|
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
machine->vtbl->GetVRDEServer(machine, &VRDxServer);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
|
|
|
if (VRDxServer) {
|
|
|
|
VRDxServer->vtbl->GetEnabled(VRDxServer, &VRDxEnabled);
|
|
|
|
if (VRDxEnabled) {
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
totalPresent++;
|
|
|
|
|
|
|
|
if ((VIR_REALLOC_N(def->graphics, totalPresent) >= 0) &&
|
|
|
|
(VIR_ALLOC(def->graphics[def->ngraphics]) >= 0)) {
|
|
|
|
PRUnichar *netAddressUtf16 = NULL;
|
|
|
|
char *netAddressUtf8 = NULL;
|
|
|
|
PRBool allowMultiConnection = PR_FALSE;
|
|
|
|
PRBool reuseSingleConnection = PR_FALSE;
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
PRUint32 VRDPport = 0;
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->GetPort(VRDxServer, &VRDPport);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (VRDPport) {
|
|
|
|
def->graphics[def->ngraphics]->data.rdp.port = VRDPport;
|
2010-12-27 22:35:30 +00:00
|
|
|
#elif VBOX_API_VERSION < 4000 /* 3001 <= VBOX_API_VERSION < 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *VRDPport = NULL;
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->GetPorts(VRDxServer, &VRDPport);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (VRDPport) {
|
|
|
|
/* even if vbox supports mutilpe ports, single port for now here */
|
|
|
|
def->graphics[def->ngraphics]->data.rdp.port = PRUnicharToInt(VRDPport);
|
|
|
|
VBOX_UTF16_FREE(VRDPport);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
PRUnichar *VRDEPortsKey = NULL;
|
|
|
|
PRUnichar *VRDEPortsValue = NULL;
|
|
|
|
VBOX_UTF8_TO_UTF16("TCP/Ports", &VRDEPortsKey);
|
|
|
|
VRDxServer->vtbl->GetVRDEProperty(VRDxServer, VRDEPortsKey, &VRDEPortsValue);
|
|
|
|
VBOX_UTF16_FREE(VRDEPortsKey);
|
|
|
|
if (VRDEPortsValue) {
|
|
|
|
/* even if vbox supports mutilpe ports, single port for now here */
|
|
|
|
def->graphics[def->ngraphics]->data.rdp.port = PRUnicharToInt(VRDEPortsValue);
|
|
|
|
VBOX_UTF16_FREE(VRDEPortsValue);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
|
|
|
def->graphics[def->ngraphics]->data.rdp.autoport = 1;
|
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->graphics[def->ngraphics]->type = VIR_DOMAIN_GRAPHICS_TYPE_RDP;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION >= 4000
|
|
|
|
PRUnichar *VRDENetAddressKey = NULL;
|
|
|
|
VBOX_UTF8_TO_UTF16("TCP/Address", &VRDENetAddressKey);
|
|
|
|
VRDxServer->vtbl->GetVRDEProperty(VRDxServer, VRDENetAddressKey, &netAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(VRDENetAddressKey);
|
|
|
|
#else /* VBOX_API_VERSION < 4000 */
|
|
|
|
VRDxServer->vtbl->GetNetAddress(VRDxServer, &netAddressUtf16);
|
|
|
|
#endif /* VBOX_API_VERSION < 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
if (netAddressUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(netAddressUtf16, &netAddressUtf8);
|
|
|
|
if (STRNEQ(netAddressUtf8, ""))
|
|
|
|
def->graphics[def->ngraphics]->data.rdp.listenAddr = strdup(netAddressUtf8);
|
|
|
|
VBOX_UTF16_FREE(netAddressUtf16);
|
|
|
|
VBOX_UTF8_FREE(netAddressUtf8);
|
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->GetAllowMultiConnection(VRDxServer, &allowMultiConnection);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (allowMultiConnection) {
|
|
|
|
def->graphics[def->ngraphics]->data.rdp.multiUser = 1;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->GetReuseSingleConnection(VRDxServer, &reuseSingleConnection);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (reuseSingleConnection) {
|
|
|
|
def->graphics[def->ngraphics]->data.rdp.replaceUser = 1;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->ngraphics++;
|
2009-09-07 15:44:12 +00:00
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_RELEASE(VRDxServer);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
/* dump IDE hdds if present */
|
|
|
|
VBOX_UTF8_TO_UTF16(hddBus, &hddBusUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->ndisks = 0;
|
|
|
|
machine->vtbl->GetHardDisk(machine, hddBusUtf16, 0, 0, &hardDiskPM);
|
|
|
|
if (hardDiskPM)
|
|
|
|
def->ndisks++;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetHardDisk(machine, hddBusUtf16, 0, 1, &hardDiskPS);
|
|
|
|
if (hardDiskPS)
|
|
|
|
def->ndisks++;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetHardDisk(machine, hddBusUtf16, 1, 1, &hardDiskSS);
|
|
|
|
if (hardDiskSS)
|
|
|
|
def->ndisks++;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(hddBusUtf16);
|
|
|
|
|
|
|
|
if ((def->ndisks > 0) && (VIR_ALLOC_N(def->disks, def->ndisks) >= 0)) {
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
|
|
if (VIR_ALLOC(def->disks[i]) >= 0) {
|
|
|
|
def->disks[i]->device = VIR_DOMAIN_DISK_DEVICE_DISK;
|
|
|
|
def->disks[i]->bus = VIR_DOMAIN_DISK_BUS_IDE;
|
|
|
|
def->disks[i]->type = VIR_DOMAIN_DISK_TYPE_FILE;
|
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hardDiskPM) {
|
|
|
|
PRUnichar *hddlocationUtf16 = NULL;
|
|
|
|
char *hddlocation = NULL;
|
|
|
|
PRUint32 hddType = HardDiskType_Normal;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
hardDiskPM->vtbl->imedium.GetLocation((IMedium *)hardDiskPM, &hddlocationUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(hddlocationUtf16, &hddlocation);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
hardDiskPM->vtbl->GetType(hardDiskPM, &hddType);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hddType == HardDiskType_Immutable)
|
|
|
|
def->disks[hddNum]->readonly = 1;
|
|
|
|
def->disks[hddNum]->src = strdup(hddlocation);
|
|
|
|
def->disks[hddNum]->dst = strdup("hda");
|
|
|
|
hddNum++;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(hddlocation);
|
|
|
|
VBOX_UTF16_FREE(hddlocationUtf16);
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDiskPM);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hardDiskPS) {
|
|
|
|
PRUnichar *hddlocationUtf16 = NULL;
|
|
|
|
char *hddlocation = NULL;
|
|
|
|
PRUint32 hddType = HardDiskType_Normal;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
hardDiskPS->vtbl->imedium.GetLocation((IMedium *)hardDiskPS, &hddlocationUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(hddlocationUtf16, &hddlocation);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
hardDiskPS->vtbl->GetType(hardDiskPS, &hddType);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hddType == HardDiskType_Immutable)
|
|
|
|
def->disks[hddNum]->readonly = 1;
|
|
|
|
def->disks[hddNum]->src = strdup(hddlocation);
|
|
|
|
def->disks[hddNum]->dst = strdup("hdb");
|
|
|
|
hddNum++;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(hddlocation);
|
|
|
|
VBOX_UTF16_FREE(hddlocationUtf16);
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDiskPS);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hardDiskSS) {
|
|
|
|
PRUnichar *hddlocationUtf16 = NULL;
|
|
|
|
char *hddlocation = NULL;
|
|
|
|
PRUint32 hddType = HardDiskType_Normal;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
hardDiskSS->vtbl->imedium.GetLocation((IMedium *)hardDiskSS, &hddlocationUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(hddlocationUtf16, &hddlocation);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
hardDiskSS->vtbl->GetType(hardDiskSS, &hddType);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hddType == HardDiskType_Immutable)
|
|
|
|
def->disks[hddNum]->readonly = 1;
|
|
|
|
def->disks[hddNum]->src = strdup(hddlocation);
|
|
|
|
def->disks[hddNum]->dst = strdup("hdd");
|
|
|
|
hddNum++;
|
|
|
|
|
|
|
|
VBOX_UTF8_FREE(hddlocation);
|
|
|
|
VBOX_UTF16_FREE(hddlocationUtf16);
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDiskSS);
|
|
|
|
}
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
/* dump IDE hdds if present */
|
|
|
|
|
|
|
|
bool error = false;
|
|
|
|
int diskCount = 0;
|
|
|
|
PRUint32 maxPortPerInst[StorageBus_Floppy + 1] = {};
|
|
|
|
PRUint32 maxSlotPerPort[StorageBus_Floppy + 1] = {};
|
|
|
|
def->ndisks = 0;
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&mediumAttachments, machine, machine->vtbl->GetMediumAttachments);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
/* get the number of attachments */
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < mediumAttachments.count; i++) {
|
|
|
|
IMediumAttachment *imediumattach = mediumAttachments.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
if (imediumattach) {
|
|
|
|
IMedium *medium = NULL;
|
|
|
|
|
|
|
|
imediumattach->vtbl->GetMedium(imediumattach, &medium);
|
|
|
|
if (medium) {
|
|
|
|
def->ndisks++;
|
|
|
|
VBOX_RELEASE(medium);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Allocate mem, if fails return error */
|
|
|
|
if (VIR_ALLOC_N(def->disks, def->ndisks) >= 0) {
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
|
|
if (VIR_ALLOC(def->disks[i]) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
error = true;
|
|
|
|
break;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
error = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!error)
|
|
|
|
error = !vboxGetMaxPortSlotValues(data->vboxObj, maxPortPerInst, maxSlotPerPort);
|
|
|
|
|
|
|
|
/* get the attachment details here */
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < mediumAttachments.count && diskCount < def->ndisks && !error; i++) {
|
|
|
|
IMediumAttachment *imediumattach = mediumAttachments.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
IStorageController *storageController = NULL;
|
|
|
|
PRUnichar *storageControllerName = NULL;
|
|
|
|
PRUint32 deviceType = DeviceType_Null;
|
|
|
|
PRUint32 storageBus = StorageBus_Null;
|
|
|
|
PRBool readOnly = PR_FALSE;
|
|
|
|
IMedium *medium = NULL;
|
|
|
|
PRUnichar *mediumLocUtf16 = NULL;
|
|
|
|
char *mediumLocUtf8 = NULL;
|
|
|
|
PRUint32 deviceInst = 0;
|
|
|
|
PRInt32 devicePort = 0;
|
|
|
|
PRInt32 deviceSlot = 0;
|
|
|
|
|
|
|
|
if (!imediumattach)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
imediumattach->vtbl->GetMedium(imediumattach, &medium);
|
|
|
|
if (!medium)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
imediumattach->vtbl->GetController(imediumattach, &storageControllerName);
|
|
|
|
if (!storageControllerName) {
|
|
|
|
VBOX_RELEASE(medium);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
machine->vtbl->GetStorageControllerByName(machine,
|
|
|
|
storageControllerName,
|
|
|
|
&storageController);
|
|
|
|
VBOX_UTF16_FREE(storageControllerName);
|
|
|
|
if (!storageController) {
|
|
|
|
VBOX_RELEASE(medium);
|
|
|
|
continue;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
medium->vtbl->GetLocation(medium, &mediumLocUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(mediumLocUtf16, &mediumLocUtf8);
|
|
|
|
VBOX_UTF16_FREE(mediumLocUtf16);
|
|
|
|
def->disks[diskCount]->src = strdup(mediumLocUtf8);
|
|
|
|
VBOX_UTF8_FREE(mediumLocUtf8);
|
|
|
|
|
|
|
|
if (!(def->disks[diskCount]->src)) {
|
|
|
|
VBOX_RELEASE(medium);
|
|
|
|
VBOX_RELEASE(storageController);
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
error = true;
|
|
|
|
break;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
storageController->vtbl->GetBus(storageController, &storageBus);
|
|
|
|
if (storageBus == StorageBus_IDE) {
|
|
|
|
def->disks[diskCount]->bus = VIR_DOMAIN_DISK_BUS_IDE;
|
|
|
|
} else if (storageBus == StorageBus_SATA) {
|
|
|
|
def->disks[diskCount]->bus = VIR_DOMAIN_DISK_BUS_SATA;
|
|
|
|
} else if (storageBus == StorageBus_SCSI) {
|
|
|
|
def->disks[diskCount]->bus = VIR_DOMAIN_DISK_BUS_SCSI;
|
|
|
|
} else if (storageBus == StorageBus_Floppy) {
|
|
|
|
def->disks[diskCount]->bus = VIR_DOMAIN_DISK_BUS_FDC;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
imediumattach->vtbl->GetType(imediumattach, &deviceType);
|
|
|
|
if (deviceType == DeviceType_HardDisk)
|
|
|
|
def->disks[diskCount]->device = VIR_DOMAIN_DISK_DEVICE_DISK;
|
|
|
|
else if (deviceType == DeviceType_Floppy)
|
|
|
|
def->disks[diskCount]->device = VIR_DOMAIN_DISK_DEVICE_FLOPPY;
|
|
|
|
else if (deviceType == DeviceType_DVD)
|
|
|
|
def->disks[diskCount]->device = VIR_DOMAIN_DISK_DEVICE_CDROM;
|
|
|
|
|
|
|
|
imediumattach->vtbl->GetPort(imediumattach, &devicePort);
|
|
|
|
imediumattach->vtbl->GetDevice(imediumattach, &deviceSlot);
|
2009-12-04 14:18:45 +00:00
|
|
|
def->disks[diskCount]->dst = vboxGenerateMediumName(storageBus,
|
2009-12-04 13:49:45 +00:00
|
|
|
deviceInst,
|
|
|
|
devicePort,
|
|
|
|
deviceSlot,
|
|
|
|
maxPortPerInst,
|
|
|
|
maxSlotPerPort);
|
|
|
|
if (!def->disks[diskCount]->dst) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-04-03 19:17:00 +00:00
|
|
|
_("Could not generate medium name for the disk "
|
|
|
|
"at: controller instance:%u, port:%d, slot:%d"),
|
2009-12-04 13:49:45 +00:00
|
|
|
deviceInst, devicePort, deviceSlot);
|
|
|
|
VBOX_RELEASE(medium);
|
|
|
|
VBOX_RELEASE(storageController);
|
|
|
|
error = true;
|
|
|
|
break;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
medium->vtbl->GetReadOnly(medium, &readOnly);
|
|
|
|
if (readOnly == PR_TRUE)
|
|
|
|
def->disks[diskCount]->readonly = 1;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->disks[diskCount]->type = VIR_DOMAIN_DISK_TYPE_FILE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(medium);
|
|
|
|
VBOX_RELEASE(storageController);
|
|
|
|
diskCount++;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&mediumAttachments);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* cleanup on error */
|
|
|
|
if (error) {
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
|
|
|
VIR_FREE(def->disks[i]);
|
|
|
|
}
|
|
|
|
VIR_FREE(def->disks);
|
|
|
|
def->ndisks = 0;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* dump network cards if present */
|
|
|
|
def->nnets = 0;
|
|
|
|
/* Get which network cards are enabled */
|
|
|
|
for (i = 0; i < netAdpCnt; i++) {
|
|
|
|
INetworkAdapter *adapter = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetNetworkAdapter(machine, i, &adapter);
|
|
|
|
if (adapter) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
adapter->vtbl->GetEnabled(adapter, &enabled);
|
|
|
|
if (enabled) {
|
|
|
|
def->nnets++;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(adapter);
|
|
|
|
}
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Allocate memory for the networkcards which are enabled */
|
|
|
|
if ((def->nnets > 0) && (VIR_ALLOC_N(def->nets, def->nnets) >= 0)) {
|
|
|
|
for (i = 0; i < def->nnets; i++) {
|
|
|
|
if (VIR_ALLOC(def->nets[i]) >= 0) {
|
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Now get the details about the network cards here */
|
|
|
|
for (i = 0;(netAdpIncCnt < def->nnets) && (i < netAdpCnt); i++) {
|
|
|
|
INetworkAdapter *adapter = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetNetworkAdapter(machine, i, &adapter);
|
|
|
|
if (adapter) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
adapter->vtbl->GetEnabled(adapter, &enabled);
|
|
|
|
if (enabled) {
|
|
|
|
PRUint32 attachmentType = NetworkAttachmentType_Null;
|
|
|
|
PRUint32 adapterType = NetworkAdapterType_Null;
|
|
|
|
PRUnichar *MACAddressUtf16 = NULL;
|
|
|
|
char *MACAddress = NULL;
|
|
|
|
char macaddr[VIR_MAC_STRING_BUFLEN] = {0};
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
adapter->vtbl->GetAttachmentType(adapter, &attachmentType);
|
|
|
|
if (attachmentType == NetworkAttachmentType_NAT) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->nets[netAdpIncCnt]->type = VIR_DOMAIN_NET_TYPE_USER;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
} else if (attachmentType == NetworkAttachmentType_Bridged) {
|
|
|
|
PRUnichar *hostIntUtf16 = NULL;
|
|
|
|
char *hostInt = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->nets[netAdpIncCnt]->type = VIR_DOMAIN_NET_TYPE_BRIDGE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
adapter->vtbl->GetHostInterface(adapter, &hostIntUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(hostIntUtf16, &hostInt);
|
|
|
|
def->nets[netAdpIncCnt]->data.bridge.brname = strdup(hostInt);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(hostInt);
|
|
|
|
VBOX_UTF16_FREE(hostIntUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
} else if (attachmentType == NetworkAttachmentType_Internal) {
|
|
|
|
PRUnichar *intNetUtf16 = NULL;
|
|
|
|
char *intNet = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->nets[netAdpIncCnt]->type = VIR_DOMAIN_NET_TYPE_INTERNAL;
|
|
|
|
|
|
|
|
adapter->vtbl->GetInternalNetwork(adapter, &intNetUtf16);
|
|
|
|
|
|
|
|
VBOX_UTF16_TO_UTF8(intNetUtf16, &intNet);
|
|
|
|
def->nets[netAdpIncCnt]->data.internal.name = strdup(intNet);
|
|
|
|
|
|
|
|
VBOX_UTF8_FREE(intNet);
|
|
|
|
VBOX_UTF16_FREE(intNetUtf16);
|
|
|
|
|
|
|
|
} else if (attachmentType == NetworkAttachmentType_HostOnly) {
|
|
|
|
PRUnichar *hostIntUtf16 = NULL;
|
|
|
|
char *hostInt = NULL;
|
|
|
|
|
|
|
|
def->nets[netAdpIncCnt]->type = VIR_DOMAIN_NET_TYPE_NETWORK;
|
|
|
|
|
|
|
|
adapter->vtbl->GetHostInterface(adapter, &hostIntUtf16);
|
|
|
|
|
|
|
|
VBOX_UTF16_TO_UTF8(hostIntUtf16, &hostInt);
|
|
|
|
def->nets[netAdpIncCnt]->data.network.name = strdup(hostInt);
|
|
|
|
|
|
|
|
VBOX_UTF8_FREE(hostInt);
|
|
|
|
VBOX_UTF16_FREE(hostIntUtf16);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
/* default to user type i.e. NAT in VirtualBox if this
|
|
|
|
* dump is ever used to create a machine.
|
|
|
|
*/
|
|
|
|
def->nets[netAdpIncCnt]->type = VIR_DOMAIN_NET_TYPE_USER;
|
|
|
|
}
|
|
|
|
|
|
|
|
adapter->vtbl->GetAdapterType(adapter, &adapterType);
|
|
|
|
if (adapterType == NetworkAdapterType_Am79C970A) {
|
|
|
|
def->nets[netAdpIncCnt]->model = strdup("Am79C970A");
|
|
|
|
} else if (adapterType == NetworkAdapterType_Am79C973) {
|
|
|
|
def->nets[netAdpIncCnt]->model = strdup("Am79C973");
|
|
|
|
} else if (adapterType == NetworkAdapterType_I82540EM) {
|
|
|
|
def->nets[netAdpIncCnt]->model = strdup("82540EM");
|
|
|
|
} else if (adapterType == NetworkAdapterType_I82545EM) {
|
|
|
|
def->nets[netAdpIncCnt]->model = strdup("82545EM");
|
|
|
|
} else if (adapterType == NetworkAdapterType_I82543GC) {
|
|
|
|
def->nets[netAdpIncCnt]->model = strdup("82543GC");
|
|
|
|
#if VBOX_API_VERSION >= 3001
|
|
|
|
} else if (adapterType == NetworkAdapterType_Virtio) {
|
|
|
|
def->nets[netAdpIncCnt]->model = strdup("virtio");
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
adapter->vtbl->GetMACAddress(adapter, &MACAddressUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(MACAddressUtf16, &MACAddress);
|
|
|
|
snprintf(macaddr, VIR_MAC_STRING_BUFLEN,
|
|
|
|
"%c%c:%c%c:%c%c:%c%c:%c%c:%c%c",
|
|
|
|
MACAddress[0], MACAddress[1], MACAddress[2], MACAddress[3],
|
|
|
|
MACAddress[4], MACAddress[5], MACAddress[6], MACAddress[7],
|
|
|
|
MACAddress[8], MACAddress[9], MACAddress[10], MACAddress[11]);
|
|
|
|
|
|
|
|
/* XXX some real error handling here some day ... */
|
|
|
|
if (virParseMacAddr(macaddr, def->nets[netAdpIncCnt]->mac) < 0)
|
|
|
|
{}
|
|
|
|
|
|
|
|
netAdpIncCnt++;
|
|
|
|
|
|
|
|
VBOX_UTF16_FREE(MACAddressUtf16);
|
|
|
|
VBOX_UTF8_FREE(MACAddress);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(adapter);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* dump sound card if active */
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Set def->nsounds to one as VirtualBox currently supports
|
|
|
|
* only one sound card
|
|
|
|
*/
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAudioAdapter(machine, &audioAdapter);
|
|
|
|
if (audioAdapter) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
|
|
|
|
|
|
|
audioAdapter->vtbl->GetEnabled(audioAdapter, &enabled);
|
|
|
|
if (enabled) {
|
|
|
|
PRUint32 audioController = AudioControllerType_AC97;
|
|
|
|
|
|
|
|
def->nsounds = 1;
|
|
|
|
if (VIR_ALLOC_N(def->sounds, def->nsounds) >= 0) {
|
|
|
|
if (VIR_ALLOC(def->sounds[0]) >= 0) {
|
|
|
|
audioAdapter->vtbl->GetAudioController(audioAdapter, &audioController);
|
|
|
|
if (audioController == AudioControllerType_SB16) {
|
|
|
|
def->sounds[0]->model = VIR_DOMAIN_SOUND_MODEL_SB16;
|
|
|
|
} else if (audioController == AudioControllerType_AC97) {
|
|
|
|
def->sounds[0]->model = VIR_DOMAIN_SOUND_MODEL_AC97;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
VIR_FREE(def->sounds);
|
|
|
|
def->nsounds = 0;
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
def->nsounds = 0;
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
VBOX_RELEASE(audioAdapter);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
/* dump CDROM/DVD if the drive is attached and has DVD/CD in it */
|
|
|
|
machine->vtbl->GetDVDDrive(machine, &dvdDrive);
|
|
|
|
if (dvdDrive) {
|
|
|
|
PRUint32 state = DriveState_Null;
|
|
|
|
|
|
|
|
dvdDrive->vtbl->GetState(dvdDrive, &state);
|
|
|
|
if (state == DriveState_ImageMounted) {
|
|
|
|
IDVDImage *dvdImage = NULL;
|
|
|
|
|
|
|
|
dvdDrive->vtbl->GetImage(dvdDrive, &dvdImage);
|
|
|
|
if (dvdImage) {
|
|
|
|
PRUnichar *locationUtf16 = NULL;
|
|
|
|
char *location = NULL;
|
|
|
|
|
|
|
|
dvdImage->vtbl->imedium.GetLocation((IMedium *)dvdImage, &locationUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(locationUtf16, &location);
|
|
|
|
|
|
|
|
def->ndisks++;
|
|
|
|
if (VIR_REALLOC_N(def->disks, def->ndisks) >= 0) {
|
|
|
|
if (VIR_ALLOC(def->disks[def->ndisks - 1]) >= 0) {
|
|
|
|
def->disks[def->ndisks - 1]->device = VIR_DOMAIN_DISK_DEVICE_CDROM;
|
|
|
|
def->disks[def->ndisks - 1]->bus = VIR_DOMAIN_DISK_BUS_IDE;
|
|
|
|
def->disks[def->ndisks - 1]->type = VIR_DOMAIN_DISK_TYPE_FILE;
|
|
|
|
def->disks[def->ndisks - 1]->readonly = 1;
|
|
|
|
def->disks[def->ndisks - 1]->src = strdup(location);
|
|
|
|
def->disks[def->ndisks - 1]->dst = strdup("hdc");
|
2009-04-17 16:09:07 +00:00
|
|
|
} else {
|
2009-12-04 13:49:45 +00:00
|
|
|
def->ndisks--;
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
} else {
|
2009-12-04 13:49:45 +00:00
|
|
|
def->ndisks--;
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_UTF8_FREE(location);
|
|
|
|
VBOX_UTF16_FREE(locationUtf16);
|
|
|
|
VBOX_MEDIUM_RELEASE(dvdImage);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(dvdDrive);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* dump Floppy if the drive is attached and has floppy in it */
|
|
|
|
machine->vtbl->GetFloppyDrive(machine, &floppyDrive);
|
|
|
|
if (floppyDrive) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
|
|
|
|
|
|
|
floppyDrive->vtbl->GetEnabled(floppyDrive, &enabled);
|
|
|
|
if (enabled) {
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state = DriveState_Null;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
floppyDrive->vtbl->GetState(floppyDrive, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
if (state == DriveState_ImageMounted) {
|
2009-12-04 13:49:45 +00:00
|
|
|
IFloppyImage *floppyImage = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
floppyDrive->vtbl->GetImage(floppyDrive, &floppyImage);
|
|
|
|
if (floppyImage) {
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUnichar *locationUtf16 = NULL;
|
|
|
|
char *location = NULL;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
floppyImage->vtbl->imedium.GetLocation((IMedium *)floppyImage, &locationUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(locationUtf16, &location);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
def->ndisks++;
|
|
|
|
if (VIR_REALLOC_N(def->disks, def->ndisks) >= 0) {
|
|
|
|
if (VIR_ALLOC(def->disks[def->ndisks - 1]) >= 0) {
|
2009-12-04 13:49:45 +00:00
|
|
|
def->disks[def->ndisks - 1]->device = VIR_DOMAIN_DISK_DEVICE_FLOPPY;
|
|
|
|
def->disks[def->ndisks - 1]->bus = VIR_DOMAIN_DISK_BUS_FDC;
|
2009-04-17 16:09:07 +00:00
|
|
|
def->disks[def->ndisks - 1]->type = VIR_DOMAIN_DISK_TYPE_FILE;
|
2009-12-04 13:49:45 +00:00
|
|
|
def->disks[def->ndisks - 1]->readonly = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
def->disks[def->ndisks - 1]->src = strdup(location);
|
2009-12-04 13:49:45 +00:00
|
|
|
def->disks[def->ndisks - 1]->dst = strdup("fda");
|
2009-04-17 16:09:07 +00:00
|
|
|
} else {
|
|
|
|
def->ndisks--;
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
def->ndisks--;
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(location);
|
|
|
|
VBOX_UTF16_FREE(locationUtf16);
|
|
|
|
VBOX_MEDIUM_RELEASE(floppyImage);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(floppyDrive);
|
|
|
|
}
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
|
|
|
|
|
|
|
/* dump serial port if active */
|
|
|
|
def->nserials = 0;
|
|
|
|
/* Get which serial ports are enabled/active */
|
|
|
|
for (i = 0; i < serialPortCount; i++) {
|
|
|
|
ISerialPort *serialPort = NULL;
|
|
|
|
|
|
|
|
machine->vtbl->GetSerialPort(machine, i, &serialPort);
|
|
|
|
if (serialPort) {
|
2009-04-17 16:09:07 +00:00
|
|
|
PRBool enabled = PR_FALSE;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
serialPort->vtbl->GetEnabled(serialPort, &enabled);
|
2009-04-17 16:09:07 +00:00
|
|
|
if (enabled) {
|
2009-12-04 13:49:45 +00:00
|
|
|
def->nserials++;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(serialPort);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Allocate memory for the serial ports which are enabled */
|
|
|
|
if ((def->nserials > 0) && (VIR_ALLOC_N(def->serials, def->nserials) >= 0)) {
|
|
|
|
for (i = 0; i < def->nserials; i++) {
|
|
|
|
if (VIR_ALLOC(def->serials[i]) >= 0) {
|
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Now get the details about the serial ports here */
|
|
|
|
for (i = 0;(serialPortIncCount < def->nserials) && (i < serialPortCount); i++) {
|
|
|
|
ISerialPort *serialPort = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetSerialPort(machine, i, &serialPort);
|
|
|
|
if (serialPort) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
serialPort->vtbl->GetEnabled(serialPort, &enabled);
|
|
|
|
if (enabled) {
|
|
|
|
PRUint32 hostMode = PortMode_Disconnected;
|
|
|
|
PRUint32 IOBase = 0;
|
|
|
|
PRUint32 IRQ = 0;
|
|
|
|
PRUnichar *pathUtf16 = NULL;
|
|
|
|
char *path = NULL;
|
|
|
|
|
|
|
|
serialPort->vtbl->GetHostMode(serialPort, &hostMode);
|
|
|
|
if (hostMode == PortMode_HostPipe) {
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
def->serials[serialPortIncCount]->source.type = VIR_DOMAIN_CHR_TYPE_PIPE;
|
2009-12-04 13:49:45 +00:00
|
|
|
} else if (hostMode == PortMode_HostDevice) {
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
def->serials[serialPortIncCount]->source.type = VIR_DOMAIN_CHR_TYPE_DEV;
|
2009-12-04 13:49:45 +00:00
|
|
|
#if VBOX_API_VERSION >= 3000
|
|
|
|
} else if (hostMode == PortMode_RawFile) {
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
def->serials[serialPortIncCount]->source.type = VIR_DOMAIN_CHR_TYPE_FILE;
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif /* VBOX_API_VERSION >= 3000 */
|
|
|
|
} else {
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
def->serials[serialPortIncCount]->source.type = VIR_DOMAIN_CHR_TYPE_NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-07-13 19:05:34 +00:00
|
|
|
def->serials[serialPortIncCount]->deviceType = VIR_DOMAIN_CHR_DEVICE_TYPE_SERIAL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
serialPort->vtbl->GetIRQ(serialPort, &IRQ);
|
|
|
|
serialPort->vtbl->GetIOBase(serialPort, &IOBase);
|
|
|
|
if ((IRQ == 4) && (IOBase == 1016)) {
|
|
|
|
def->serials[serialPortIncCount]->target.port = 0;
|
|
|
|
} else if ((IRQ == 3) && (IOBase == 760)) {
|
|
|
|
def->serials[serialPortIncCount]->target.port = 1;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
serialPort->vtbl->GetPath(serialPort, &pathUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (pathUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(pathUtf16, &path);
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
def->serials[serialPortIncCount]->source.data.file.path = strdup(path);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
serialPortIncCount++;
|
|
|
|
|
|
|
|
VBOX_UTF16_FREE(pathUtf16);
|
|
|
|
VBOX_UTF8_FREE(path);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(serialPort);
|
|
|
|
}
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* dump parallel ports if active */
|
|
|
|
def->nparallels = 0;
|
|
|
|
/* Get which parallel ports are enabled/active */
|
|
|
|
for (i = 0; i < parallelPortCount; i++) {
|
|
|
|
IParallelPort *parallelPort = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetParallelPort(machine, i, ¶llelPort);
|
|
|
|
if (parallelPort) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
parallelPort->vtbl->GetEnabled(parallelPort, &enabled);
|
|
|
|
if (enabled) {
|
|
|
|
def->nparallels++;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(parallelPort);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Allocate memory for the parallel ports which are enabled */
|
|
|
|
if ((def->nparallels > 0) && (VIR_ALLOC_N(def->parallels, def->nparallels) >= 0)) {
|
|
|
|
for (i = 0; i < def->nparallels; i++) {
|
|
|
|
if (VIR_ALLOC(def->parallels[i]) >= 0) {
|
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Now get the details about the parallel ports here */
|
|
|
|
for (i = 0;(parallelPortIncCount < def->nparallels) && (i < parallelPortCount); i++) {
|
|
|
|
IParallelPort *parallelPort = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetParallelPort(machine, i, ¶llelPort);
|
|
|
|
if (parallelPort) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
parallelPort->vtbl->GetEnabled(parallelPort, &enabled);
|
|
|
|
if (enabled) {
|
|
|
|
PRUint32 IOBase = 0;
|
|
|
|
PRUint32 IRQ = 0;
|
|
|
|
PRUnichar *pathUtf16 = NULL;
|
|
|
|
char *path = NULL;
|
|
|
|
|
|
|
|
parallelPort->vtbl->GetIRQ(parallelPort, &IRQ);
|
|
|
|
parallelPort->vtbl->GetIOBase(parallelPort, &IOBase);
|
|
|
|
if ((IRQ == 7) && (IOBase == 888)) {
|
|
|
|
def->parallels[parallelPortIncCount]->target.port = 0;
|
|
|
|
} else if ((IRQ == 5) && (IOBase == 632)) {
|
|
|
|
def->parallels[parallelPortIncCount]->target.port = 1;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
def->parallels[parallelPortIncCount]->source.type = VIR_DOMAIN_CHR_TYPE_FILE;
|
2010-07-13 19:05:34 +00:00
|
|
|
def->parallels[parallelPortIncCount]->deviceType = VIR_DOMAIN_CHR_DEVICE_TYPE_PARALLEL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
parallelPort->vtbl->GetPath(parallelPort, &pathUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(pathUtf16, &path);
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
def->parallels[parallelPortIncCount]->source.data.file.path = strdup(path);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
parallelPortIncCount++;
|
|
|
|
|
|
|
|
VBOX_UTF16_FREE(pathUtf16);
|
|
|
|
VBOX_UTF8_FREE(path);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(parallelPort);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* dump USB devices/filters if active */
|
|
|
|
def->nhostdevs = 0;
|
|
|
|
machine->vtbl->GetUSBController(machine, &USBController);
|
|
|
|
if (USBController) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
USBController->vtbl->GetEnabled(USBController, &enabled);
|
|
|
|
if (enabled) {
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray deviceFilters = VBOX_ARRAY_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&deviceFilters, USBController,
|
|
|
|
USBController->vtbl->GetDeviceFilters);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
if (deviceFilters.count > 0) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* check if the filters are active and then only
|
|
|
|
* alloc mem and set def->nhostdevs
|
|
|
|
*/
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for(i = 0; i < deviceFilters.count; i++) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool active = PR_FALSE;
|
2010-12-16 22:05:48 +00:00
|
|
|
IUSBDeviceFilter *deviceFilter = deviceFilters.items[i];
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
deviceFilter->vtbl->GetActive(deviceFilter, &active);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (active) {
|
|
|
|
def->nhostdevs++;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (def->nhostdevs > 0) {
|
|
|
|
/* Alloc mem needed for the filters now */
|
|
|
|
if (VIR_ALLOC_N(def->hostdevs, def->nhostdevs) >= 0) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for(i = 0; (USBFilterCount < def->nhostdevs) || (i < deviceFilters.count); i++) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool active = PR_FALSE;
|
2010-12-16 22:05:48 +00:00
|
|
|
IUSBDeviceFilter *deviceFilter = deviceFilters.items[i];
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
deviceFilter->vtbl->GetActive(deviceFilter, &active);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (active) {
|
|
|
|
if (VIR_ALLOC(def->hostdevs[USBFilterCount]) >= 0) {
|
|
|
|
PRUnichar *vendorIdUtf16 = NULL;
|
|
|
|
char *vendorIdUtf8 = NULL;
|
|
|
|
unsigned vendorId = 0;
|
|
|
|
PRUnichar *productIdUtf16 = NULL;
|
|
|
|
char *productIdUtf8 = NULL;
|
|
|
|
unsigned productId = 0;
|
|
|
|
char *endptr = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->hostdevs[USBFilterCount]->mode =
|
|
|
|
VIR_DOMAIN_HOSTDEV_MODE_SUBSYS;
|
|
|
|
def->hostdevs[USBFilterCount]->source.subsys.type =
|
|
|
|
VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
deviceFilter->vtbl->GetVendorId(deviceFilter, &vendorIdUtf16);
|
|
|
|
deviceFilter->vtbl->GetProductId(deviceFilter, &productIdUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(vendorIdUtf16, &vendorIdUtf8);
|
|
|
|
VBOX_UTF16_TO_UTF8(productIdUtf16, &productIdUtf8);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
vendorId = strtol(vendorIdUtf8, &endptr, 16);
|
|
|
|
productId = strtol(productIdUtf8, &endptr, 16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->hostdevs[USBFilterCount]->source.subsys.u.usb.vendor = vendorId;
|
|
|
|
def->hostdevs[USBFilterCount]->source.subsys.u.usb.product = productId;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(vendorIdUtf16);
|
|
|
|
VBOX_UTF8_FREE(vendorIdUtf8);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(productIdUtf16);
|
|
|
|
VBOX_UTF8_FREE(productIdUtf8);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
USBFilterCount++;
|
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
} else
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Cleanup */
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&deviceFilters);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
VBOX_RELEASE(USBController);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
/* all done so set gotAllABoutDef and pass def to virDomainDefFormat
|
|
|
|
* to generate XML for it
|
|
|
|
*/
|
|
|
|
gotAllABoutDef = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
|
|
|
machine = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (gotAllABoutDef == 0)
|
2010-02-09 18:58:01 +00:00
|
|
|
ret = virDomainDefFormat(def, flags);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
cleanup:
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
virDomainDefFree(def);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxListDefinedDomains(virConnectPtr conn, char ** const names, int maxnames) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
char *machineName = NULL;
|
|
|
|
PRUnichar *machineNameUtf16 = NULL;
|
|
|
|
PRUint32 state;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
int i, j;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get list of Defined Domains, rc=%08x"),
|
|
|
|
(unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
ret = 0;
|
|
|
|
for (i = 0, j = 0; (i < machines.count) && (j < maxnames); i++) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (machine) {
|
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
|
|
|
if ( (state < MachineState_FirstOnline)
|
|
|
|
|| (state > MachineState_LastOnline) ) {
|
|
|
|
machine->vtbl->GetName(machine, &machineNameUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(machineNameUtf16, &machineName);
|
2010-12-16 22:33:14 +00:00
|
|
|
names[j] = strdup(machineName);
|
|
|
|
VBOX_UTF16_FREE(machineNameUtf16);
|
|
|
|
VBOX_UTF8_FREE(machineName);
|
|
|
|
if (!names[j]) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
for ( ; j >= 0 ; j--)
|
|
|
|
VIR_FREE(names[j]);
|
|
|
|
ret = -1;
|
|
|
|
goto cleanup;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-16 22:33:14 +00:00
|
|
|
j++;
|
2009-12-04 13:49:45 +00:00
|
|
|
ret++;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxNumOfDefinedDomains(virConnectPtr conn) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state = MachineState_Null;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
int i;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get number of Defined Domains, rc=%08x"),
|
|
|
|
(unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
ret = 0;
|
|
|
|
for (i = 0; i < machines.count; ++i) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (machine) {
|
|
|
|
PRBool isAccessible = PR_FALSE;
|
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
|
|
|
if ( (state < MachineState_FirstOnline)
|
|
|
|
|| (state > MachineState_LastOnline) ) {
|
|
|
|
ret++;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-08-18 23:20:07 +00:00
|
|
|
|
|
|
|
static int
|
2010-12-27 22:35:30 +00:00
|
|
|
vboxStartMachine(virDomainPtr dom, int i, IMachine *machine,
|
|
|
|
vboxIID *iid ATTRIBUTE_UNUSED /* >= 4.0 */)
|
2010-08-18 23:20:07 +00:00
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
|
|
|
int vrdpPresent = 0;
|
|
|
|
int sdlPresent = 0;
|
|
|
|
int guiPresent = 0;
|
|
|
|
char *guiDisplay = NULL;
|
|
|
|
char *sdlDisplay = NULL;
|
|
|
|
PRUnichar *keyTypeUtf16 = NULL;
|
|
|
|
PRUnichar *valueTypeUtf16 = NULL;
|
|
|
|
char *valueTypeUtf8 = NULL;
|
|
|
|
PRUnichar *keyDislpayUtf16 = NULL;
|
|
|
|
PRUnichar *valueDisplayUtf16 = NULL;
|
|
|
|
char *valueDisplayUtf8 = NULL;
|
|
|
|
IProgress *progress = NULL;
|
|
|
|
PRUnichar *env = NULL;
|
|
|
|
PRUnichar *sessionType = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Type", &keyTypeUtf16);
|
|
|
|
machine->vtbl->GetExtraData(machine, keyTypeUtf16, &valueTypeUtf16);
|
|
|
|
VBOX_UTF16_FREE(keyTypeUtf16);
|
|
|
|
|
|
|
|
if (valueTypeUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(valueTypeUtf16, &valueTypeUtf8);
|
|
|
|
VBOX_UTF16_FREE(valueTypeUtf16);
|
|
|
|
|
|
|
|
if ( STREQ(valueTypeUtf8, "sdl") || STREQ(valueTypeUtf8, "gui") ) {
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Display", &keyDislpayUtf16);
|
|
|
|
machine->vtbl->GetExtraData(machine, keyDislpayUtf16,
|
|
|
|
&valueDisplayUtf16);
|
|
|
|
VBOX_UTF16_FREE(keyDislpayUtf16);
|
|
|
|
|
|
|
|
if (valueDisplayUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(valueDisplayUtf16, &valueDisplayUtf8);
|
|
|
|
VBOX_UTF16_FREE(valueDisplayUtf16);
|
|
|
|
|
|
|
|
if (strlen(valueDisplayUtf8) <= 0) {
|
|
|
|
VBOX_UTF8_FREE(valueDisplayUtf8);
|
|
|
|
valueDisplayUtf8 = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STREQ(valueTypeUtf8, "sdl")) {
|
|
|
|
sdlPresent = 1;
|
|
|
|
if (valueDisplayUtf8) {
|
|
|
|
sdlDisplay = strdup(valueDisplayUtf8);
|
|
|
|
if (sdlDisplay == NULL) {
|
|
|
|
virReportOOMError();
|
|
|
|
/* just don't go to cleanup yet as it is ok to have
|
|
|
|
* sdlDisplay as NULL and we check it below if it
|
|
|
|
* exist and then only use it there
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STREQ(valueTypeUtf8, "gui")) {
|
|
|
|
guiPresent = 1;
|
|
|
|
if (valueDisplayUtf8) {
|
|
|
|
guiDisplay = strdup(valueDisplayUtf8);
|
|
|
|
if (guiDisplay == NULL) {
|
|
|
|
virReportOOMError();
|
|
|
|
/* just don't go to cleanup yet as it is ok to have
|
|
|
|
* guiDisplay as NULL and we check it below if it
|
|
|
|
* exist and then only use it there
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STREQ(valueTypeUtf8, "vrdp")) {
|
|
|
|
vrdpPresent = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!vrdpPresent && !sdlPresent && !guiPresent) {
|
|
|
|
/* if nothing is selected it means either the machine xml
|
|
|
|
* file is really old or some values are missing so fallback
|
|
|
|
*/
|
|
|
|
guiPresent = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
VBOX_UTF8_FREE(valueTypeUtf8);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
guiPresent = 1;
|
|
|
|
}
|
|
|
|
if (valueDisplayUtf8)
|
|
|
|
VBOX_UTF8_FREE(valueDisplayUtf8);
|
|
|
|
|
|
|
|
if (guiPresent) {
|
|
|
|
if (guiDisplay) {
|
2010-08-18 23:31:39 +00:00
|
|
|
char *displayutf8;
|
|
|
|
if (virAsprintf(&displayutf8, "DISPLAY=%s", guiDisplay) < 0)
|
|
|
|
virReportOOMError();
|
|
|
|
else {
|
|
|
|
VBOX_UTF8_TO_UTF16(displayutf8, &env);
|
|
|
|
VIR_FREE(displayutf8);
|
|
|
|
}
|
2010-08-18 23:20:07 +00:00
|
|
|
VIR_FREE(guiDisplay);
|
|
|
|
}
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("gui", &sessionType);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sdlPresent) {
|
|
|
|
if (sdlDisplay) {
|
2010-08-18 23:31:39 +00:00
|
|
|
char *displayutf8;
|
|
|
|
if (virAsprintf(&displayutf8, "DISPLAY=%s", sdlDisplay) < 0)
|
|
|
|
virReportOOMError();
|
|
|
|
else {
|
|
|
|
VBOX_UTF8_TO_UTF16(displayutf8, &env);
|
|
|
|
VIR_FREE(displayutf8);
|
|
|
|
}
|
2010-08-18 23:20:07 +00:00
|
|
|
VIR_FREE(sdlDisplay);
|
|
|
|
}
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("sdl", &sessionType);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vrdpPresent) {
|
|
|
|
VBOX_UTF8_TO_UTF16("vrdp", &sessionType);
|
|
|
|
}
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2010-08-18 23:20:07 +00:00
|
|
|
rc = data->vboxObj->vtbl->OpenRemoteSession(data->vboxObj,
|
|
|
|
data->vboxSession,
|
2010-12-23 16:25:56 +00:00
|
|
|
iid->value,
|
2010-08-18 23:20:07 +00:00
|
|
|
sessionType,
|
|
|
|
env,
|
|
|
|
&progress );
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = machine->vtbl->LaunchVMProcess(machine, data->vboxSession,
|
|
|
|
sessionType, env, &progress);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
|
|
|
|
2010-08-18 23:20:07 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
2010-12-27 22:35:30 +00:00
|
|
|
_("OpenRemoteSession/LaunchVMProcess failed, domain can't be started"));
|
2010-08-18 23:20:07 +00:00
|
|
|
ret = -1;
|
|
|
|
} else {
|
|
|
|
PRBool completed = 0;
|
|
|
|
#if VBOX_API_VERSION == 2002
|
|
|
|
nsresult resultCode;
|
|
|
|
#else
|
|
|
|
PRInt32 resultCode;
|
|
|
|
#endif
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
rc = progress->vtbl->GetCompleted(progress, &completed);
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
/* error */
|
|
|
|
ret = -1;
|
|
|
|
}
|
|
|
|
progress->vtbl->GetResultCode(progress, &resultCode);
|
|
|
|
if (NS_FAILED(resultCode)) {
|
|
|
|
/* error */
|
|
|
|
ret = -1;
|
|
|
|
} else {
|
|
|
|
/* all ok set the domid */
|
|
|
|
dom->id = i + 1;
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
VBOX_RELEASE(progress);
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2010-08-18 23:20:07 +00:00
|
|
|
|
|
|
|
VBOX_UTF16_FREE(env);
|
|
|
|
VBOX_UTF16_FREE(sessionType);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-06-10 15:55:36 +00:00
|
|
|
static int vboxDomainCreateWithFlags(virDomainPtr dom, unsigned int flags) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machines = VBOX_ARRAY_INITIALIZER;
|
2010-12-23 16:25:56 +00:00
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN] = {0};
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
|
|
|
int i = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-06-10 15:55:36 +00:00
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
if (!dom->name) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-04-03 19:17:00 +00:00
|
|
|
_("Error while reading the domain name"));
|
2009-04-17 16:09:07 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&machines, data->vboxObj, data->vboxObj->vtbl->GetMachines);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Could not get list of machines, rc=%08x"), (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < machines.count; ++i) {
|
|
|
|
IMachine *machine = machines.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
PRBool isAccessible = PR_FALSE;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!machine)
|
|
|
|
continue;
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->GetAccessible(machine, &isAccessible);
|
|
|
|
if (isAccessible) {
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = machine->vtbl->GetId(machine, &iid.value);
|
|
|
|
if (NS_FAILED(rc))
|
2009-12-04 13:49:45 +00:00
|
|
|
continue;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDToUUID(&iid, uuid);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (memcmp(dom->uuid, uuid, VIR_UUID_BUFLEN) == 0) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 state = MachineState_Null;
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
|
|
|
|
|
|
|
if ( (state == MachineState_PoweredOff) ||
|
|
|
|
(state == MachineState_Saved) ||
|
|
|
|
(state == MachineState_Aborted) ) {
|
2010-12-23 16:25:56 +00:00
|
|
|
ret = vboxStartMachine(dom, i, machine, &iid);
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_FAILED, "%s",
|
|
|
|
_("machine is not in poweroff|saved|"
|
|
|
|
"aborted state, so couldn't start it"));
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = -1;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (ret != -1)
|
|
|
|
break;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Do the cleanup and take care you dont leak any memory */
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&machines);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
cleanup:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-06-10 15:55:36 +00:00
|
|
|
static int vboxDomainCreate(virDomainPtr dom) {
|
|
|
|
return vboxDomainCreateWithFlags(dom, 0);
|
|
|
|
}
|
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
static void
|
|
|
|
vboxSetBootDeviceOrder(virDomainDefPtr def, vboxGlobalData *data,
|
|
|
|
IMachine *machine)
|
|
|
|
{
|
|
|
|
ISystemProperties *systemProperties = NULL;
|
|
|
|
PRUint32 maxBootPosition = 0;
|
|
|
|
int i = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("def->os.type %s", def->os.type);
|
|
|
|
VIR_DEBUG("def->os.arch %s", def->os.arch);
|
|
|
|
VIR_DEBUG("def->os.machine %s", def->os.machine);
|
|
|
|
VIR_DEBUG("def->os.nBootDevs %d", def->os.nBootDevs);
|
|
|
|
VIR_DEBUG("def->os.bootDevs[0] %d", def->os.bootDevs[0]);
|
|
|
|
VIR_DEBUG("def->os.bootDevs[1] %d", def->os.bootDevs[1]);
|
|
|
|
VIR_DEBUG("def->os.bootDevs[2] %d", def->os.bootDevs[2]);
|
|
|
|
VIR_DEBUG("def->os.bootDevs[3] %d", def->os.bootDevs[3]);
|
|
|
|
VIR_DEBUG("def->os.init %s", def->os.init);
|
|
|
|
VIR_DEBUG("def->os.kernel %s", def->os.kernel);
|
|
|
|
VIR_DEBUG("def->os.initrd %s", def->os.initrd);
|
|
|
|
VIR_DEBUG("def->os.cmdline %s", def->os.cmdline);
|
|
|
|
VIR_DEBUG("def->os.root %s", def->os.root);
|
|
|
|
VIR_DEBUG("def->os.loader %s", def->os.loader);
|
|
|
|
VIR_DEBUG("def->os.bootloader %s", def->os.bootloader);
|
|
|
|
VIR_DEBUG("def->os.bootloaderArgs %s", def->os.bootloaderArgs);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->GetSystemProperties(data->vboxObj, &systemProperties);
|
|
|
|
if (systemProperties) {
|
|
|
|
systemProperties->vtbl->GetMaxBootPosition(systemProperties,
|
|
|
|
&maxBootPosition);
|
|
|
|
VBOX_RELEASE(systemProperties);
|
|
|
|
systemProperties = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
/* Clear the defaults first */
|
|
|
|
for (i = 0; i < maxBootPosition; i++) {
|
|
|
|
machine->vtbl->SetBootOrder(machine, i+1, DeviceType_Null);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
for (i = 0; (i < def->os.nBootDevs) && (i < maxBootPosition); i++) {
|
|
|
|
PRUint32 device = DeviceType_Null;
|
|
|
|
|
|
|
|
if (def->os.bootDevs[i] == VIR_DOMAIN_BOOT_FLOPPY) {
|
|
|
|
device = DeviceType_Floppy;
|
|
|
|
} else if (def->os.bootDevs[i] == VIR_DOMAIN_BOOT_CDROM) {
|
|
|
|
device = DeviceType_DVD;
|
|
|
|
} else if (def->os.bootDevs[i] == VIR_DOMAIN_BOOT_DISK) {
|
|
|
|
device = DeviceType_HardDisk;
|
|
|
|
} else if (def->os.bootDevs[i] == VIR_DOMAIN_BOOT_NET) {
|
|
|
|
device = DeviceType_Network;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetBootOrder(machine, i+1, device);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
static void
|
|
|
|
vboxAttachDrives(virDomainDefPtr def, vboxGlobalData *data, IMachine *machine)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
nsresult rc;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->ndisks == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < def->ndisks; i++) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("disk(%d) type: %d", i, def->disks[i]->type);
|
|
|
|
VIR_DEBUG("disk(%d) device: %d", i, def->disks[i]->device);
|
|
|
|
VIR_DEBUG("disk(%d) bus: %d", i, def->disks[i]->bus);
|
|
|
|
VIR_DEBUG("disk(%d) src: %s", i, def->disks[i]->src);
|
|
|
|
VIR_DEBUG("disk(%d) dst: %s", i, def->disks[i]->dst);
|
|
|
|
VIR_DEBUG("disk(%d) driverName: %s", i, def->disks[i]->driverName);
|
|
|
|
VIR_DEBUG("disk(%d) driverType: %s", i, def->disks[i]->driverType);
|
|
|
|
VIR_DEBUG("disk(%d) cachemode: %d", i, def->disks[i]->cachemode);
|
|
|
|
VIR_DEBUG("disk(%d) readonly: %s", i, (def->disks[i]->readonly
|
2010-08-31 18:01:45 +00:00
|
|
|
? "True" : "False"));
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("disk(%d) shared: %s", i, (def->disks[i]->shared
|
2010-08-31 18:01:45 +00:00
|
|
|
? "True" : "False"));
|
|
|
|
|
|
|
|
if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
|
|
|
if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_FILE &&
|
|
|
|
def->disks[i]->src != NULL) {
|
|
|
|
IDVDDrive *dvdDrive = NULL;
|
|
|
|
/* Currently CDROM/DVD Drive is always IDE
|
|
|
|
* Secondary Master so neglecting the following
|
|
|
|
* parameters:
|
|
|
|
* def->disks[i]->bus
|
|
|
|
* def->disks[i]->dst
|
|
|
|
*/
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->GetDVDDrive(machine, &dvdDrive);
|
|
|
|
if (dvdDrive) {
|
|
|
|
IDVDImage *dvdImage = NULL;
|
|
|
|
PRUnichar *dvdfileUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID dvduuid = VBOX_IID_INITIALIZER;
|
|
|
|
vboxIID dvdemptyuuid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(def->disks[i]->src, &dvdfileUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->FindDVDImage(data->vboxObj,
|
|
|
|
dvdfileUtf16, &dvdImage);
|
|
|
|
if (!dvdImage) {
|
|
|
|
data->vboxObj->vtbl->OpenDVDImage(data->vboxObj,
|
|
|
|
dvdfileUtf16,
|
2010-12-23 16:25:56 +00:00
|
|
|
dvdemptyuuid.value,
|
2010-08-31 18:01:45 +00:00
|
|
|
&dvdImage);
|
|
|
|
}
|
|
|
|
if (dvdImage) {
|
|
|
|
rc = dvdImage->vtbl->imedium.GetId((IMedium *)dvdImage,
|
2010-12-23 16:25:56 +00:00
|
|
|
&dvduuid.value);
|
2010-08-31 18:01:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't get the uuid of the file to "
|
|
|
|
"be attached to cdrom: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
} else {
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = dvdDrive->vtbl->MountImage(dvdDrive, dvduuid.value);
|
2010-08-31 18:01:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not attach the file to cdrom: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
} else {
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("CD/DVDImage UUID:", dvduuid.value);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
|
|
|
|
VBOX_MEDIUM_RELEASE(dvdImage);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&dvduuid);
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(dvdfileUtf16);
|
|
|
|
VBOX_RELEASE(dvdDrive);
|
|
|
|
}
|
|
|
|
} else if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
|
|
|
}
|
|
|
|
} else if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_DISK) {
|
|
|
|
if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_FILE &&
|
|
|
|
def->disks[i]->src != NULL) {
|
|
|
|
IHardDisk *hardDisk = NULL;
|
|
|
|
PRUnichar *hddfileUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID hdduuid = VBOX_IID_INITIALIZER;
|
2010-08-31 18:01:45 +00:00
|
|
|
PRUnichar *hddEmpty = NULL;
|
|
|
|
/* Current Limitation: Harddisk can't be connected to
|
|
|
|
* Secondary Master as Secondary Master is always used
|
|
|
|
* for CD/DVD Drive, so don't connect the harddisk if it
|
|
|
|
* is requested to be connected to Secondary master
|
|
|
|
*/
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(def->disks[i]->src, &hddfileUtf16);
|
|
|
|
VBOX_UTF8_TO_UTF16("", &hddEmpty);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->FindHardDisk(data->vboxObj, hddfileUtf16,
|
|
|
|
&hardDisk);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (!hardDisk) {
|
2010-03-09 18:22:22 +00:00
|
|
|
# if VBOX_API_VERSION == 2002
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->OpenHardDisk(data->vboxObj,
|
|
|
|
hddfileUtf16,
|
|
|
|
AccessMode_ReadWrite,
|
|
|
|
&hardDisk);
|
2010-03-09 18:22:22 +00:00
|
|
|
# else
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->OpenHardDisk(data->vboxObj,
|
|
|
|
hddfileUtf16,
|
|
|
|
AccessMode_ReadWrite,
|
|
|
|
0,
|
|
|
|
hddEmpty,
|
|
|
|
0,
|
|
|
|
hddEmpty,
|
|
|
|
&hardDisk);
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (hardDisk) {
|
|
|
|
rc = hardDisk->vtbl->imedium.GetId((IMedium *)hardDisk,
|
2010-12-23 16:25:56 +00:00
|
|
|
&hdduuid.value);
|
2010-08-31 18:01:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't get the uuid of the file to be "
|
|
|
|
"attached as harddisk: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
} else {
|
|
|
|
if (def->disks[i]->readonly) {
|
|
|
|
hardDisk->vtbl->SetType(hardDisk,
|
|
|
|
HardDiskType_Immutable);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("setting harddisk to readonly");
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (!def->disks[i]->readonly) {
|
|
|
|
hardDisk->vtbl->SetType(hardDisk,
|
|
|
|
HardDiskType_Normal);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("setting harddisk type to normal");
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
|
|
|
if (def->disks[i]->bus == VIR_DOMAIN_DISK_BUS_IDE) {
|
|
|
|
if (STREQ(def->disks[i]->dst, "hdc")) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("Not connecting harddisk to hdc as hdc"
|
2010-08-31 18:01:45 +00:00
|
|
|
" is taken by CD/DVD Drive");
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-08-31 18:01:45 +00:00
|
|
|
PRInt32 channel = 0;
|
|
|
|
PRInt32 device = 0;
|
|
|
|
PRUnichar *hddcnameUtf16 = NULL;
|
|
|
|
|
|
|
|
char *hddcname = strdup("IDE");
|
|
|
|
VBOX_UTF8_TO_UTF16(hddcname, &hddcnameUtf16);
|
|
|
|
VIR_FREE(hddcname);
|
|
|
|
|
|
|
|
if (STREQ(def->disks[i]->dst, "hda")) {
|
|
|
|
channel = 0;
|
|
|
|
device = 0;
|
|
|
|
} else if (STREQ(def->disks[i]->dst, "hdb")) {
|
|
|
|
channel = 0;
|
|
|
|
device = 1;
|
|
|
|
} else if (STREQ(def->disks[i]->dst, "hdd")) {
|
|
|
|
channel = 1;
|
|
|
|
device = 1;
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
|
|
|
|
rc = machine->vtbl->AttachHardDisk(machine,
|
2010-12-23 16:25:56 +00:00
|
|
|
hdduuid.value,
|
2010-08-31 18:01:45 +00:00
|
|
|
hddcnameUtf16,
|
|
|
|
channel,
|
|
|
|
device);
|
|
|
|
VBOX_UTF16_FREE(hddcnameUtf16);
|
|
|
|
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not attach the file as "
|
|
|
|
"harddisk: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
} else {
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("Attached HDD with UUID", hdduuid.value);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_MEDIUM_RELEASE(hardDisk);
|
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&hdduuid);
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(hddEmpty);
|
|
|
|
VBOX_UTF16_FREE(hddfileUtf16);
|
|
|
|
} else if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
|
|
|
}
|
|
|
|
} else if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
|
|
|
if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_FILE &&
|
|
|
|
def->disks[i]->src != NULL) {
|
|
|
|
IFloppyDrive *floppyDrive;
|
|
|
|
machine->vtbl->GetFloppyDrive(machine, &floppyDrive);
|
|
|
|
if (floppyDrive) {
|
|
|
|
rc = floppyDrive->vtbl->SetEnabled(floppyDrive, 1);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
IFloppyImage *floppyImage = NULL;
|
|
|
|
PRUnichar *fdfileUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID fduuid = VBOX_IID_INITIALIZER;
|
|
|
|
vboxIID fdemptyuuid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(def->disks[i]->src, &fdfileUtf16);
|
|
|
|
rc = data->vboxObj->vtbl->FindFloppyImage(data->vboxObj,
|
|
|
|
fdfileUtf16,
|
|
|
|
&floppyImage);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (!floppyImage) {
|
|
|
|
data->vboxObj->vtbl->OpenFloppyImage(data->vboxObj,
|
|
|
|
fdfileUtf16,
|
2010-12-23 16:25:56 +00:00
|
|
|
fdemptyuuid.value,
|
2010-08-31 18:01:45 +00:00
|
|
|
&floppyImage);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (floppyImage) {
|
|
|
|
rc = floppyImage->vtbl->imedium.GetId((IMedium *)floppyImage,
|
2010-12-23 16:25:56 +00:00
|
|
|
&fduuid.value);
|
2010-08-31 18:01:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't get the uuid of the file to "
|
|
|
|
"be attached to floppy drive: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
} else {
|
|
|
|
rc = floppyDrive->vtbl->MountImage(floppyDrive,
|
2010-12-23 16:25:56 +00:00
|
|
|
fduuid.value);
|
2010-08-31 18:01:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not attach the file to "
|
|
|
|
"floppy drive: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
} else {
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("floppyImage UUID", fduuid.value);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_MEDIUM_RELEASE(floppyImage);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&fduuid);
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(fdfileUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_RELEASE(floppyDrive);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
2010-08-31 18:01:45 +00:00
|
|
|
PRUint32 maxPortPerInst[StorageBus_Floppy + 1] = {};
|
|
|
|
PRUint32 maxSlotPerPort[StorageBus_Floppy + 1] = {};
|
|
|
|
PRUnichar *storageCtlName = NULL;
|
|
|
|
bool error = false;
|
|
|
|
|
|
|
|
/* get the max port/slots/etc for the given storage bus */
|
|
|
|
error = !vboxGetMaxPortSlotValues(data->vboxObj, maxPortPerInst,
|
|
|
|
maxSlotPerPort);
|
|
|
|
|
|
|
|
/* add a storage controller for the mediums to be attached */
|
|
|
|
/* this needs to change when multiple controller are supported for
|
|
|
|
* ver > 3.1 */
|
2009-12-04 13:49:45 +00:00
|
|
|
{
|
2010-08-31 18:01:45 +00:00
|
|
|
IStorageController *storageCtl = NULL;
|
|
|
|
PRUnichar *sName = NULL;
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("IDE Controller", &sName);
|
|
|
|
machine->vtbl->AddStorageController(machine,
|
|
|
|
sName,
|
|
|
|
StorageBus_IDE,
|
|
|
|
&storageCtl);
|
|
|
|
VBOX_UTF16_FREE(sName);
|
|
|
|
VBOX_RELEASE(storageCtl);
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("SATA Controller", &sName);
|
|
|
|
machine->vtbl->AddStorageController(machine,
|
|
|
|
sName,
|
|
|
|
StorageBus_SATA,
|
|
|
|
&storageCtl);
|
|
|
|
VBOX_UTF16_FREE(sName);
|
|
|
|
VBOX_RELEASE(storageCtl);
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("SCSI Controller", &sName);
|
|
|
|
machine->vtbl->AddStorageController(machine,
|
|
|
|
sName,
|
|
|
|
StorageBus_SCSI,
|
|
|
|
&storageCtl);
|
|
|
|
VBOX_UTF16_FREE(sName);
|
|
|
|
VBOX_RELEASE(storageCtl);
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("Floppy Controller", &sName);
|
|
|
|
machine->vtbl->AddStorageController(machine,
|
|
|
|
sName,
|
|
|
|
StorageBus_Floppy,
|
|
|
|
&storageCtl);
|
|
|
|
VBOX_UTF16_FREE(sName);
|
|
|
|
VBOX_RELEASE(storageCtl);
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
for (i = 0; i < def->ndisks && !error; i++) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("disk(%d) type: %d", i, def->disks[i]->type);
|
|
|
|
VIR_DEBUG("disk(%d) device: %d", i, def->disks[i]->device);
|
|
|
|
VIR_DEBUG("disk(%d) bus: %d", i, def->disks[i]->bus);
|
|
|
|
VIR_DEBUG("disk(%d) src: %s", i, def->disks[i]->src);
|
|
|
|
VIR_DEBUG("disk(%d) dst: %s", i, def->disks[i]->dst);
|
|
|
|
VIR_DEBUG("disk(%d) driverName: %s", i, def->disks[i]->driverName);
|
|
|
|
VIR_DEBUG("disk(%d) driverType: %s", i, def->disks[i]->driverType);
|
|
|
|
VIR_DEBUG("disk(%d) cachemode: %d", i, def->disks[i]->cachemode);
|
|
|
|
VIR_DEBUG("disk(%d) readonly: %s", i, (def->disks[i]->readonly
|
2010-08-31 18:01:45 +00:00
|
|
|
? "True" : "False"));
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("disk(%d) shared: %s", i, (def->disks[i]->shared
|
2010-08-31 18:01:45 +00:00
|
|
|
? "True" : "False"));
|
|
|
|
|
|
|
|
if (def->disks[i]->type == VIR_DOMAIN_DISK_TYPE_FILE &&
|
|
|
|
def->disks[i]->src != NULL) {
|
|
|
|
IMedium *medium = NULL;
|
|
|
|
PRUnichar *mediumUUID = NULL;
|
|
|
|
PRUnichar *mediumFileUtf16 = NULL;
|
|
|
|
PRUint32 storageBus = StorageBus_Null;
|
|
|
|
PRUint32 deviceType = DeviceType_Null;
|
2010-12-27 22:35:30 +00:00
|
|
|
PRUint32 accessMode = AccessMode_ReadOnly;
|
2010-08-31 18:01:45 +00:00
|
|
|
PRInt32 deviceInst = 0;
|
|
|
|
PRInt32 devicePort = 0;
|
|
|
|
PRInt32 deviceSlot = 0;
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16(def->disks[i]->src, &mediumFileUtf16);
|
|
|
|
|
|
|
|
if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_DISK) {
|
|
|
|
deviceType = DeviceType_HardDisk;
|
2010-12-27 22:35:30 +00:00
|
|
|
accessMode = AccessMode_ReadWrite;
|
|
|
|
# if VBOX_API_VERSION < 4000
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->FindHardDisk(data->vboxObj,
|
|
|
|
mediumFileUtf16, &medium);
|
2010-12-27 22:35:30 +00:00
|
|
|
# endif
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
|
|
|
deviceType = DeviceType_DVD;
|
2010-12-27 22:35:30 +00:00
|
|
|
accessMode = AccessMode_ReadOnly;
|
|
|
|
# if VBOX_API_VERSION < 4000
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->FindDVDImage(data->vboxObj,
|
|
|
|
mediumFileUtf16, &medium);
|
2010-12-27 22:35:30 +00:00
|
|
|
# endif
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
|
|
|
deviceType = DeviceType_Floppy;
|
2010-12-27 22:35:30 +00:00
|
|
|
accessMode = AccessMode_ReadWrite;
|
|
|
|
# if VBOX_API_VERSION < 4000
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->FindFloppyImage(data->vboxObj,
|
|
|
|
mediumFileUtf16, &medium);
|
2010-12-27 22:35:30 +00:00
|
|
|
# endif
|
2010-08-31 18:01:45 +00:00
|
|
|
} else {
|
|
|
|
VBOX_UTF16_FREE(mediumFileUtf16);
|
|
|
|
continue;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
# if VBOX_API_VERSION >= 4000
|
|
|
|
data->vboxObj->vtbl->FindMedium(data->vboxObj, mediumFileUtf16,
|
|
|
|
deviceType, &medium);
|
|
|
|
# endif
|
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (!medium) {
|
|
|
|
PRUnichar *mediumEmpty = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16("", &mediumEmpty);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
# if VBOX_API_VERSION < 4000
|
2009-12-04 13:49:45 +00:00
|
|
|
if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_DISK) {
|
2010-09-03 15:39:23 +00:00
|
|
|
rc = data->vboxObj->vtbl->OpenHardDisk(data->vboxObj,
|
|
|
|
mediumFileUtf16,
|
|
|
|
AccessMode_ReadWrite,
|
|
|
|
false,
|
|
|
|
mediumEmpty,
|
|
|
|
false,
|
|
|
|
mediumEmpty,
|
|
|
|
&medium);
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->disks[i]->device ==
|
|
|
|
VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
2010-09-03 15:39:23 +00:00
|
|
|
rc = data->vboxObj->vtbl->OpenDVDImage(data->vboxObj,
|
|
|
|
mediumFileUtf16,
|
|
|
|
mediumEmpty,
|
|
|
|
&medium);
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->disks[i]->device ==
|
|
|
|
VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
2010-09-03 15:39:23 +00:00
|
|
|
rc = data->vboxObj->vtbl->OpenFloppyImage(data->vboxObj,
|
|
|
|
mediumFileUtf16,
|
|
|
|
mediumEmpty,
|
|
|
|
&medium);
|
|
|
|
} else {
|
|
|
|
rc = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
# else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = data->vboxObj->vtbl->OpenMedium(data->vboxObj,
|
|
|
|
mediumFileUtf16,
|
|
|
|
deviceType, accessMode,
|
|
|
|
&medium);
|
|
|
|
# endif /* VBOX_API_VERSION >= 4000 */
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(mediumEmpty);
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (!medium) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Failed to attach the following disk/dvd/floppy "
|
|
|
|
"to the machine: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
VBOX_UTF16_FREE(mediumFileUtf16);
|
|
|
|
continue;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
rc = medium->vtbl->GetId(medium, &mediumUUID);
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't get the uuid of the file to be attached "
|
|
|
|
"as harddisk/dvd/floppy: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
VBOX_RELEASE(medium);
|
|
|
|
VBOX_UTF16_FREE(mediumFileUtf16);
|
|
|
|
continue;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_DISK) {
|
|
|
|
if (def->disks[i]->readonly) {
|
|
|
|
medium->vtbl->SetType(medium, MediumType_Immutable);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("setting harddisk to immutable");
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (!def->disks[i]->readonly) {
|
|
|
|
medium->vtbl->SetType(medium, MediumType_Normal);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("setting harddisk type to normal");
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->disks[i]->bus == VIR_DOMAIN_DISK_BUS_IDE) {
|
|
|
|
VBOX_UTF8_TO_UTF16("IDE Controller", &storageCtlName);
|
|
|
|
storageBus = StorageBus_IDE;
|
|
|
|
} else if (def->disks[i]->bus == VIR_DOMAIN_DISK_BUS_SATA) {
|
|
|
|
VBOX_UTF8_TO_UTF16("SATA Controller", &storageCtlName);
|
|
|
|
storageBus = StorageBus_SATA;
|
|
|
|
} else if (def->disks[i]->bus == VIR_DOMAIN_DISK_BUS_SCSI) {
|
|
|
|
VBOX_UTF8_TO_UTF16("SCSI Controller", &storageCtlName);
|
|
|
|
storageBus = StorageBus_SCSI;
|
|
|
|
} else if (def->disks[i]->bus == VIR_DOMAIN_DISK_BUS_FDC) {
|
|
|
|
VBOX_UTF8_TO_UTF16("Floppy Controller", &storageCtlName);
|
|
|
|
storageBus = StorageBus_Floppy;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
/* get the device details i.e instance, port and slot */
|
|
|
|
if (!vboxGetDeviceDetails(def->disks[i]->dst,
|
|
|
|
maxPortPerInst,
|
|
|
|
maxSlotPerPort,
|
|
|
|
storageBus,
|
|
|
|
&deviceInst,
|
|
|
|
&devicePort,
|
|
|
|
&deviceSlot)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't get the port/slot number of harddisk/"
|
|
|
|
"dvd/floppy to be attached: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(medium);
|
|
|
|
VBOX_UTF16_FREE(mediumUUID);
|
|
|
|
VBOX_UTF16_FREE(mediumFileUtf16);
|
2010-08-31 18:01:45 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* attach the harddisk/dvd/Floppy to the storage controller */
|
|
|
|
rc = machine->vtbl->AttachDevice(machine,
|
|
|
|
storageCtlName,
|
|
|
|
devicePort,
|
|
|
|
deviceSlot,
|
|
|
|
deviceType,
|
2010-12-27 22:35:30 +00:00
|
|
|
# if VBOX_API_VERSION < 4000
|
2010-08-31 18:01:45 +00:00
|
|
|
mediumUUID);
|
2010-12-27 22:35:30 +00:00
|
|
|
# else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
medium);
|
|
|
|
# endif /* VBOX_API_VERSION >= 4000 */
|
2010-08-31 18:01:45 +00:00
|
|
|
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not attach the file as harddisk/"
|
|
|
|
"dvd/floppy: %s, rc=%08x"),
|
|
|
|
def->disks[i]->src, (unsigned)rc);
|
|
|
|
} else {
|
|
|
|
DEBUGIID("Attached HDD/DVD/Floppy with UUID", mediumUUID);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(medium);
|
|
|
|
VBOX_UTF16_FREE(mediumUUID);
|
|
|
|
VBOX_UTF16_FREE(mediumFileUtf16);
|
|
|
|
VBOX_UTF16_FREE(storageCtlName);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
static void
|
|
|
|
vboxAttachSound(virDomainDefPtr def, IMachine *machine)
|
|
|
|
{
|
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
/* Check if def->nsounds is one as VirtualBox currently supports
|
|
|
|
* only one sound card
|
|
|
|
*/
|
|
|
|
if (def->nsounds == 1) {
|
|
|
|
IAudioAdapter *audioAdapter = NULL;
|
|
|
|
|
|
|
|
machine->vtbl->GetAudioAdapter(machine, &audioAdapter);
|
|
|
|
if (audioAdapter) {
|
|
|
|
rc = audioAdapter->vtbl->SetEnabled(audioAdapter, 1);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
if (def->sounds[0]->model == VIR_DOMAIN_SOUND_MODEL_SB16) {
|
|
|
|
audioAdapter->vtbl->SetAudioController(audioAdapter,
|
|
|
|
AudioControllerType_SB16);
|
|
|
|
} else if (def->sounds[0]->model == VIR_DOMAIN_SOUND_MODEL_AC97) {
|
|
|
|
audioAdapter->vtbl->SetAudioController(audioAdapter,
|
|
|
|
AudioControllerType_AC97);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_RELEASE(audioAdapter);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxAttachNetwork(virDomainDefPtr def, vboxGlobalData *data, IMachine *machine)
|
|
|
|
{
|
|
|
|
ISystemProperties *systemProperties = NULL;
|
|
|
|
PRUint32 networkAdapterCount = 0;
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
data->vboxObj->vtbl->GetSystemProperties(data->vboxObj, &systemProperties);
|
|
|
|
if (systemProperties) {
|
|
|
|
systemProperties->vtbl->GetNetworkAdapterCount(systemProperties,
|
|
|
|
&networkAdapterCount);
|
|
|
|
VBOX_RELEASE(systemProperties);
|
|
|
|
systemProperties = NULL;
|
|
|
|
}
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Number of Network Cards to be connected: %d", def->nnets);
|
|
|
|
VIR_DEBUG("Number of Network Cards available: %d", networkAdapterCount);
|
2010-08-31 18:01:45 +00:00
|
|
|
|
|
|
|
for (i = 0; (i < def->nnets) && (i < networkAdapterCount); i++) {
|
|
|
|
INetworkAdapter *adapter = NULL;
|
|
|
|
PRUint32 adapterType = NetworkAdapterType_Null;
|
|
|
|
char macaddr[VIR_MAC_STRING_BUFLEN] = {0};
|
|
|
|
char macaddrvbox[VIR_MAC_STRING_BUFLEN - 5] = {0};
|
|
|
|
|
|
|
|
virFormatMacAddr(def->nets[i]->mac, macaddr);
|
|
|
|
snprintf(macaddrvbox, VIR_MAC_STRING_BUFLEN - 5,
|
|
|
|
"%02X%02X%02X%02X%02X%02X",
|
|
|
|
def->nets[i]->mac[0],
|
|
|
|
def->nets[i]->mac[1],
|
|
|
|
def->nets[i]->mac[2],
|
|
|
|
def->nets[i]->mac[3],
|
|
|
|
def->nets[i]->mac[4],
|
|
|
|
def->nets[i]->mac[5]);
|
|
|
|
macaddrvbox[VIR_MAC_STRING_BUFLEN - 6] = '\0';
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("NIC(%d): Type: %d", i, def->nets[i]->type);
|
|
|
|
VIR_DEBUG("NIC(%d): Model: %s", i, def->nets[i]->model);
|
|
|
|
VIR_DEBUG("NIC(%d): Mac: %s", i, macaddr);
|
|
|
|
VIR_DEBUG("NIC(%d): ifname: %s", i, def->nets[i]->ifname);
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->nets[i]->type == VIR_DOMAIN_NET_TYPE_NETWORK) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("NIC(%d): name: %s", i, def->nets[i]->data.network.name);
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->nets[i]->type == VIR_DOMAIN_NET_TYPE_INTERNAL) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("NIC(%d): name: %s", i, def->nets[i]->data.internal.name);
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->nets[i]->type == VIR_DOMAIN_NET_TYPE_USER) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("NIC(%d): NAT.", i);
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->nets[i]->type == VIR_DOMAIN_NET_TYPE_BRIDGE) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("NIC(%d): brname: %s", i, def->nets[i]->data.bridge.brname);
|
|
|
|
VIR_DEBUG("NIC(%d): script: %s", i, def->nets[i]->data.bridge.script);
|
|
|
|
VIR_DEBUG("NIC(%d): ipaddr: %s", i, def->nets[i]->data.bridge.ipaddr);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->GetNetworkAdapter(machine, i, &adapter);
|
|
|
|
if (adapter) {
|
|
|
|
PRUnichar *MACAddress = NULL;
|
|
|
|
|
|
|
|
adapter->vtbl->SetEnabled(adapter, 1);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->nets[i]->model) {
|
|
|
|
if (STRCASEEQ(def->nets[i]->model , "Am79C970A")) {
|
|
|
|
adapterType = NetworkAdapterType_Am79C970A;
|
|
|
|
} else if (STRCASEEQ(def->nets[i]->model , "Am79C973")) {
|
|
|
|
adapterType = NetworkAdapterType_Am79C973;
|
|
|
|
} else if (STRCASEEQ(def->nets[i]->model , "82540EM")) {
|
|
|
|
adapterType = NetworkAdapterType_I82540EM;
|
|
|
|
} else if (STRCASEEQ(def->nets[i]->model , "82545EM")) {
|
|
|
|
adapterType = NetworkAdapterType_I82545EM;
|
|
|
|
} else if (STRCASEEQ(def->nets[i]->model , "82543GC")) {
|
|
|
|
adapterType = NetworkAdapterType_I82543GC;
|
2009-12-04 13:49:45 +00:00
|
|
|
#if VBOX_API_VERSION >= 3001
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (STRCASEEQ(def->nets[i]->model , "virtio")) {
|
|
|
|
adapterType = NetworkAdapterType_Virtio;
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
} else {
|
|
|
|
adapterType = NetworkAdapterType_Am79C973;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
adapter->vtbl->SetAdapterType(adapter, adapterType);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->nets[i]->type == VIR_DOMAIN_NET_TYPE_BRIDGE) {
|
|
|
|
PRUnichar *hostInterface = NULL;
|
|
|
|
/* Bridged Network */
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
adapter->vtbl->AttachToBridgedInterface(adapter);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->nets[i]->data.bridge.brname) {
|
|
|
|
VBOX_UTF8_TO_UTF16(def->nets[i]->data.bridge.brname,
|
|
|
|
&hostInterface);
|
|
|
|
adapter->vtbl->SetHostInterface(adapter, hostInterface);
|
|
|
|
VBOX_UTF16_FREE(hostInterface);
|
|
|
|
}
|
|
|
|
} else if (def->nets[i]->type == VIR_DOMAIN_NET_TYPE_INTERNAL) {
|
|
|
|
PRUnichar *internalNetwork = NULL;
|
|
|
|
/* Internal Network */
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
adapter->vtbl->AttachToInternalNetwork(adapter);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->nets[i]->data.internal.name) {
|
|
|
|
VBOX_UTF8_TO_UTF16(def->nets[i]->data.internal.name,
|
|
|
|
&internalNetwork);
|
|
|
|
adapter->vtbl->SetInternalNetwork(adapter, internalNetwork);
|
|
|
|
VBOX_UTF16_FREE(internalNetwork);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->nets[i]->type == VIR_DOMAIN_NET_TYPE_NETWORK) {
|
|
|
|
PRUnichar *hostInterface = NULL;
|
|
|
|
/* Host Only Networking (currently only vboxnet0 available
|
|
|
|
* on *nix and mac, on windows you can create and configure
|
|
|
|
* as many as you want)
|
|
|
|
*/
|
|
|
|
adapter->vtbl->AttachToHostOnlyInterface(adapter);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->nets[i]->data.network.name) {
|
|
|
|
VBOX_UTF8_TO_UTF16(def->nets[i]->data.network.name,
|
|
|
|
&hostInterface);
|
|
|
|
adapter->vtbl->SetHostInterface(adapter, hostInterface);
|
|
|
|
VBOX_UTF16_FREE(hostInterface);
|
|
|
|
}
|
|
|
|
} else if (def->nets[i]->type == VIR_DOMAIN_NET_TYPE_USER) {
|
|
|
|
/* NAT */
|
|
|
|
adapter->vtbl->AttachToNAT(adapter);
|
|
|
|
} else {
|
|
|
|
/* else always default to NAT if we don't understand
|
|
|
|
* what option is been passed to us
|
|
|
|
*/
|
|
|
|
adapter->vtbl->AttachToNAT(adapter);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16(macaddrvbox, &MACAddress);
|
|
|
|
adapter->vtbl->SetMACAddress(adapter, MACAddress);
|
|
|
|
VBOX_UTF16_FREE(MACAddress);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxAttachSerial(virDomainDefPtr def, vboxGlobalData *data, IMachine *machine)
|
|
|
|
{
|
|
|
|
ISystemProperties *systemProperties = NULL;
|
|
|
|
PRUint32 serialPortCount = 0;
|
|
|
|
int i = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->GetSystemProperties(data->vboxObj, &systemProperties);
|
|
|
|
if (systemProperties) {
|
|
|
|
systemProperties->vtbl->GetSerialPortCount(systemProperties,
|
|
|
|
&serialPortCount);
|
|
|
|
VBOX_RELEASE(systemProperties);
|
|
|
|
systemProperties = NULL;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Number of Serial Ports to be connected: %d", def->nserials);
|
|
|
|
VIR_DEBUG("Number of Serial Ports available: %d", serialPortCount);
|
2010-08-31 18:01:45 +00:00
|
|
|
for (i = 0; (i < def->nserials) && (i < serialPortCount); i++) {
|
|
|
|
ISerialPort *serialPort = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("SerialPort(%d): Type: %d", i, def->serials[i]->source.type);
|
|
|
|
VIR_DEBUG("SerialPort(%d): target.port: %d", i,
|
2010-08-31 18:01:45 +00:00
|
|
|
def->serials[i]->target.port);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->GetSerialPort(machine, i, &serialPort);
|
|
|
|
if (serialPort) {
|
|
|
|
PRUnichar *pathUtf16 = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
serialPort->vtbl->SetEnabled(serialPort, 1);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
if (def->serials[i]->source.data.file.path) {
|
|
|
|
VBOX_UTF8_TO_UTF16(def->serials[i]->source.data.file.path,
|
|
|
|
&pathUtf16);
|
2010-08-31 18:01:45 +00:00
|
|
|
serialPort->vtbl->SetPath(serialPort, pathUtf16);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For now hard code the serial ports to COM1 and COM2,
|
|
|
|
* COM1 (Base Addr: 0x3F8 (decimal: 1016), IRQ: 4)
|
|
|
|
* COM2 (Base Addr: 0x2F8 (decimal: 760), IRQ: 3)
|
|
|
|
* TODO: make this more flexible
|
|
|
|
*/
|
|
|
|
/* TODO: to improve the libvirt XMl handling so
|
|
|
|
* that def->serials[i]->target.port shows real port
|
|
|
|
* and not always start at 0
|
|
|
|
*/
|
|
|
|
if (def->serials[i]->target.port == 0) {
|
|
|
|
serialPort->vtbl->SetIRQ(serialPort, 4);
|
|
|
|
serialPort->vtbl->SetIOBase(serialPort, 1016);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG(" serialPort-%d irq: %d, iobase 0x%x, path: %s",
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
i, 4, 1016, def->serials[i]->source.data.file.path);
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (def->serials[i]->target.port == 1) {
|
|
|
|
serialPort->vtbl->SetIRQ(serialPort, 3);
|
|
|
|
serialPort->vtbl->SetIOBase(serialPort, 760);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG(" serialPort-%d irq: %d, iobase 0x%x, path: %s",
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
i, 3, 760, def->serials[i]->source.data.file.path);
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
if (def->serials[i]->source.type == VIR_DOMAIN_CHR_TYPE_DEV) {
|
2010-08-31 18:01:45 +00:00
|
|
|
serialPort->vtbl->SetHostMode(serialPort, PortMode_HostDevice);
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
} else if (def->serials[i]->source.type == VIR_DOMAIN_CHR_TYPE_PIPE) {
|
2010-08-31 18:01:45 +00:00
|
|
|
serialPort->vtbl->SetHostMode(serialPort, PortMode_HostPipe);
|
2009-12-04 13:49:45 +00:00
|
|
|
#if VBOX_API_VERSION >= 3000
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
} else if (def->serials[i]->source.type == VIR_DOMAIN_CHR_TYPE_FILE) {
|
2010-08-31 18:01:45 +00:00
|
|
|
serialPort->vtbl->SetHostMode(serialPort, PortMode_RawFile);
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif /* VBOX_API_VERSION >= 3000 */
|
2010-08-31 18:01:45 +00:00
|
|
|
} else {
|
|
|
|
serialPort->vtbl->SetHostMode(serialPort,
|
|
|
|
PortMode_Disconnected);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_RELEASE(serialPort);
|
|
|
|
if (pathUtf16) {
|
|
|
|
VBOX_UTF16_FREE(pathUtf16);
|
|
|
|
pathUtf16 = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
static void
|
|
|
|
vboxAttachParallel(virDomainDefPtr def, vboxGlobalData *data, IMachine *machine)
|
|
|
|
{
|
|
|
|
ISystemProperties *systemProperties = NULL;
|
|
|
|
PRUint32 parallelPortCount = 0;
|
|
|
|
int i = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxObj->vtbl->GetSystemProperties(data->vboxObj, &systemProperties);
|
|
|
|
if (systemProperties) {
|
|
|
|
systemProperties->vtbl->GetParallelPortCount(systemProperties,
|
|
|
|
¶llelPortCount);
|
|
|
|
VBOX_RELEASE(systemProperties);
|
|
|
|
systemProperties = NULL;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Number of Parallel Ports to be connected: %d", def->nparallels);
|
|
|
|
VIR_DEBUG("Number of Parallel Ports available: %d", parallelPortCount);
|
2010-08-31 18:01:45 +00:00
|
|
|
for (i = 0; (i < def->nparallels) && (i < parallelPortCount); i++) {
|
|
|
|
IParallelPort *parallelPort = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("ParallelPort(%d): Type: %d", i, def->parallels[i]->source.type);
|
|
|
|
VIR_DEBUG("ParallelPort(%d): target.port: %d", i,
|
2010-08-31 18:01:45 +00:00
|
|
|
def->parallels[i]->target.port);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->GetParallelPort(machine, i, ¶llelPort);
|
|
|
|
if (parallelPort) {
|
|
|
|
PRUnichar *pathUtf16 = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(def->parallels[i]->source.data.file.path, &pathUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
/* For now hard code the parallel ports to
|
|
|
|
* LPT1 (Base Addr: 0x378 (decimal: 888), IRQ: 7)
|
|
|
|
* LPT2 (Base Addr: 0x278 (decimal: 632), IRQ: 5)
|
|
|
|
* TODO: make this more flexible
|
|
|
|
*/
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
if ((def->parallels[i]->source.type == VIR_DOMAIN_CHR_TYPE_DEV) ||
|
|
|
|
(def->parallels[i]->source.type == VIR_DOMAIN_CHR_TYPE_PTY) ||
|
|
|
|
(def->parallels[i]->source.type == VIR_DOMAIN_CHR_TYPE_FILE) ||
|
|
|
|
(def->parallels[i]->source.type == VIR_DOMAIN_CHR_TYPE_PIPE)) {
|
2010-08-31 18:01:45 +00:00
|
|
|
parallelPort->vtbl->SetPath(parallelPort, pathUtf16);
|
|
|
|
if (i == 0) {
|
|
|
|
parallelPort->vtbl->SetIRQ(parallelPort, 7);
|
|
|
|
parallelPort->vtbl->SetIOBase(parallelPort, 888);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG(" parallePort-%d irq: %d, iobase 0x%x, path: %s",
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
i, 7, 888, def->parallels[i]->source.data.file.path);
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if (i == 1) {
|
|
|
|
parallelPort->vtbl->SetIRQ(parallelPort, 5);
|
|
|
|
parallelPort->vtbl->SetIOBase(parallelPort, 632);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG(" parallePort-%d irq: %d, iobase 0x%x, path: %s",
|
domain_conf: split source data out from ChrDef
This opens up the possibility of reusing the smaller ChrSourceDef
for both qemu monitor and a passthrough smartcard device.
* src/conf/domain_conf.h (_virDomainChrDef): Factor host
details...
(_virDomainChrSourceDef): ...into new struct.
(virDomainChrSourceDefFree): New prototype.
* src/conf/domain_conf.c (virDomainChrDefFree)
(virDomainChrDefParseXML, virDomainChrDefFormat): Split...
(virDomainChrSourceDefClear, virDomainChrSourceDefFree)
(virDomainChrSourceDefParseXML, virDomainChrSourceDefFormat):
...into new functions.
(virDomainChrDefParseTargetXML): Update clients to reflect type
split.
* src/vmx/vmx.c (virVMXParseSerial, virVMXParseParallel)
(virVMXFormatSerial, virVMXFormatParallel): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainOpenConsole): Likewise.
* src/xen/xend_internal.c (xenDaemonParseSxprChar)
(xenDaemonFormatSxprChr): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainDumpXML, vboxAttachSerial)
(vboxAttachParallel): Likewise.
* src/security/security_dac.c (virSecurityDACSetChardevLabel)
(virSecurityDACSetChardevCallback)
(virSecurityDACRestoreChardevLabel)
(virSecurityDACRestoreChardevCallback): Likewise.
* src/security/security_selinux.c (SELinuxSetSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback)
(SELinuxRestoreSecurityChardevLabel)
(SELinuxSetSecurityChardevCallback): Likewise.
* src/security/virt-aa-helper.c (get_files): Likewise.
* src/lxc/lxc_driver.c (lxcVmStart, lxcDomainOpenConsole):
Likewise.
* src/uml/uml_conf.c (umlBuildCommandLineChr): Likewise.
* src/uml/uml_driver.c (umlIdentifyOneChrPTY, umlIdentifyChrPTY)
(umlDomainOpenConsole): Likewise.
* src/qemu/qemu_command.c (qemuBuildChrChardevStr)
(qemuBuildChrArgStr, qemuBuildCommandLine)
(qemuParseCommandLineChr): Likewise.
* src/qemu/qemu_domain.c (qemuDomainObjPrivateXMLFormat)
(qemuDomainObjPrivateXMLParse): Likewise.
* src/qemu/qemu_cgroup.c (qemuSetupChardevCgroup): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainAttachNetDevice): Likewise.
* src/qemu/qemu_driver.c (qemudFindCharDevicePTYsMonitor)
(qemudFindCharDevicePTYs, qemuPrepareChardevDevice)
(qemuPrepareMonitorChr, qemudShutdownVMDaemon)
(qemuDomainOpenConsole): Likewise.
* src/qemu/qemu_command.h (qemuBuildChrChardevStr)
(qemuBuildChrArgStr): Delete, now that they are static.
* src/libvirt_private.syms (domain_conf.h): New exports.
* cfg.mk (useless_free_options): Update list.
* tests/qemuxml2argvtest.c (testCompareXMLToArgvFiles): Update
tests.
2011-01-07 22:45:01 +00:00
|
|
|
i, 5, 632, def->parallels[i]->source.data.file.path);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
|
|
|
|
/* like serial port, parallel port can't be enabled unless
|
|
|
|
* correct IRQ and IOBase values are specified.
|
|
|
|
*/
|
|
|
|
parallelPort->vtbl->SetEnabled(parallelPort, 1);
|
|
|
|
|
|
|
|
VBOX_RELEASE(parallelPort);
|
|
|
|
if (pathUtf16) {
|
|
|
|
VBOX_UTF16_FREE(pathUtf16);
|
|
|
|
pathUtf16 = NULL;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
vboxAttachVideo(virDomainDefPtr def, IMachine *machine)
|
|
|
|
{
|
|
|
|
if ((def->nvideos == 1) &&
|
|
|
|
(def->videos[0]->type == VIR_DOMAIN_VIDEO_TYPE_VBOX)) {
|
2011-01-28 21:03:24 +00:00
|
|
|
machine->vtbl->SetVRAMSize(machine,
|
|
|
|
VIR_DIV_UP(def->videos[0]->vram, 1024));
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetMonitorCount(machine, def->videos[0]->heads);
|
|
|
|
if (def->videos[0]->accel) {
|
|
|
|
machine->vtbl->SetAccelerate3DEnabled(machine,
|
|
|
|
def->videos[0]->accel->support3d);
|
2009-12-04 13:49:45 +00:00
|
|
|
#if VBOX_API_VERSION >= 3001
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetAccelerate2DVideoEnabled(machine,
|
|
|
|
def->videos[0]->accel->support2d);
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
2010-08-31 18:01:45 +00:00
|
|
|
} else {
|
|
|
|
machine->vtbl->SetAccelerate3DEnabled(machine, 0);
|
2009-12-04 13:49:45 +00:00
|
|
|
#if VBOX_API_VERSION >= 3001
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetAccelerate2DVideoEnabled(machine, 0);
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
static void
|
|
|
|
vboxAttachDisplay(virDomainDefPtr def, vboxGlobalData *data, IMachine *machine)
|
|
|
|
{
|
|
|
|
int vrdpPresent = 0;
|
|
|
|
int sdlPresent = 0;
|
|
|
|
int guiPresent = 0;
|
|
|
|
char *guiDisplay = NULL;
|
|
|
|
char *sdlDisplay = NULL;
|
|
|
|
int i = 0;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
for (i = 0; i < def->ngraphics; i++) {
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
|
|
|
IVRDPServer *VRDxServer = NULL;
|
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
IVRDEServer *VRDxServer = NULL;
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if ((def->graphics[i]->type == VIR_DOMAIN_GRAPHICS_TYPE_RDP) &&
|
|
|
|
(vrdpPresent == 0)) {
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
vrdpPresent = 1;
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
|
|
|
machine->vtbl->GetVRDPServer(machine, &VRDxServer);
|
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
machine->vtbl->GetVRDEServer(machine, &VRDxServer);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
|
|
|
if (VRDxServer) {
|
|
|
|
VRDxServer->vtbl->SetEnabled(VRDxServer, PR_TRUE);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("VRDP Support turned ON.");
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->graphics[i]->data.rdp.port) {
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->SetPort(VRDxServer,
|
2010-08-31 18:01:45 +00:00
|
|
|
def->graphics[i]->data.rdp.port);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("VRDP Port changed to: %d",
|
2010-08-31 18:01:45 +00:00
|
|
|
def->graphics[i]->data.rdp.port);
|
|
|
|
} else if (def->graphics[i]->data.rdp.autoport) {
|
|
|
|
/* Setting the port to 0 will reset its value to
|
|
|
|
* the default one which is 3389 currently
|
|
|
|
*/
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->SetPort(VRDxServer, 0);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("VRDP Port changed to default, which is 3389 currently");
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
#elif VBOX_API_VERSION < 4000 /* 3001 <= VBOX_API_VERSION < 4000 */
|
2010-08-31 18:01:45 +00:00
|
|
|
PRUnichar *portUtf16 = NULL;
|
|
|
|
portUtf16 = PRUnicharFromInt(def->graphics[i]->data.rdp.port);
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->SetPorts(VRDxServer, portUtf16);
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(portUtf16);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
PRUnichar *VRDEPortsKey = NULL;
|
|
|
|
PRUnichar *VRDEPortsValue = NULL;
|
|
|
|
VBOX_UTF8_TO_UTF16("TCP/Ports", &VRDEPortsKey);
|
|
|
|
VRDEPortsValue = PRUnicharFromInt(def->graphics[i]->data.rdp.port);
|
|
|
|
VRDxServer->vtbl->SetVRDEProperty(VRDxServer, VRDEPortsKey,
|
|
|
|
VRDEPortsValue);
|
|
|
|
VBOX_UTF16_FREE(VRDEPortsKey);
|
|
|
|
VBOX_UTF16_FREE(VRDEPortsValue);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->graphics[i]->data.rdp.replaceUser) {
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->SetReuseSingleConnection(VRDxServer,
|
2010-08-31 18:01:45 +00:00
|
|
|
PR_TRUE);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("VRDP set to reuse single connection");
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->graphics[i]->data.rdp.multiUser) {
|
2010-12-27 22:35:30 +00:00
|
|
|
VRDxServer->vtbl->SetAllowMultiConnection(VRDxServer,
|
2010-08-31 18:01:45 +00:00
|
|
|
PR_TRUE);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("VRDP set to allow multiple connection");
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (def->graphics[i]->data.rdp.listenAddr) {
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION >= 4000
|
|
|
|
PRUnichar *netAddressKey = NULL;
|
|
|
|
#endif
|
2010-08-31 18:01:45 +00:00
|
|
|
PRUnichar *netAddressUtf16 = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(def->graphics[i]->data.rdp.listenAddr,
|
|
|
|
&netAddressUtf16);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
|
|
|
VRDxServer->vtbl->SetNetAddress(VRDxServer,
|
2010-08-31 18:01:45 +00:00
|
|
|
netAddressUtf16);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
VBOX_UTF8_TO_UTF16("TCP/Address", &netAddressKey);
|
|
|
|
VRDxServer->vtbl->SetVRDEProperty(VRDxServer, netAddressKey,
|
|
|
|
netAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(netAddressKey);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("VRDP listen address is set to: %s",
|
2010-08-31 18:01:45 +00:00
|
|
|
def->graphics[i]->data.rdp.listenAddr);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(netAddressUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_RELEASE(VRDxServer);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if ((def->graphics[i]->type == VIR_DOMAIN_GRAPHICS_TYPE_DESKTOP) &&
|
|
|
|
(guiPresent == 0)) {
|
|
|
|
guiPresent = 1;
|
|
|
|
if (def->graphics[i]->data.desktop.display) {
|
|
|
|
guiDisplay = strdup(def->graphics[i]->data.desktop.display);
|
|
|
|
if (guiDisplay == NULL) {
|
|
|
|
virReportOOMError();
|
|
|
|
/* just don't go to cleanup yet as it is ok to have
|
|
|
|
* guiDisplay as NULL and we check it below if it
|
|
|
|
* exist and then only use it there
|
|
|
|
*/
|
2009-05-15 13:45:40 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if ((def->graphics[i]->type == VIR_DOMAIN_GRAPHICS_TYPE_SDL) &&
|
|
|
|
(sdlPresent == 0)) {
|
|
|
|
sdlPresent = 1;
|
|
|
|
if (def->graphics[i]->data.sdl.display) {
|
|
|
|
sdlDisplay = strdup(def->graphics[i]->data.sdl.display);
|
|
|
|
if (sdlDisplay == NULL) {
|
|
|
|
virReportOOMError();
|
|
|
|
/* just don't go to cleanup yet as it is ok to have
|
|
|
|
* sdlDisplay as NULL and we check it below if it
|
|
|
|
* exist and then only use it there
|
|
|
|
*/
|
2009-05-15 13:45:40 +00:00
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if ((vrdpPresent == 1) && (guiPresent == 0) && (sdlPresent == 0)) {
|
|
|
|
/* store extradata key that frontend is set to vrdp */
|
|
|
|
PRUnichar *keyTypeUtf16 = NULL;
|
|
|
|
PRUnichar *valueTypeUtf16 = NULL;
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Type", &keyTypeUtf16);
|
|
|
|
VBOX_UTF8_TO_UTF16("vrdp", &valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetExtraData(machine, keyTypeUtf16, valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(keyTypeUtf16);
|
|
|
|
VBOX_UTF16_FREE(valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
} else if ((guiPresent == 0) && (sdlPresent == 1)) {
|
|
|
|
/* store extradata key that frontend is set to sdl */
|
|
|
|
PRUnichar *keyTypeUtf16 = NULL;
|
|
|
|
PRUnichar *valueTypeUtf16 = NULL;
|
|
|
|
PRUnichar *keyDislpayUtf16 = NULL;
|
|
|
|
PRUnichar *valueDisplayUtf16 = NULL;
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Type", &keyTypeUtf16);
|
|
|
|
VBOX_UTF8_TO_UTF16("sdl", &valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetExtraData(machine, keyTypeUtf16, valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(keyTypeUtf16);
|
|
|
|
VBOX_UTF16_FREE(valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (sdlDisplay) {
|
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Display", &keyDislpayUtf16);
|
|
|
|
VBOX_UTF8_TO_UTF16(sdlDisplay, &valueDisplayUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetExtraData(machine, keyDislpayUtf16,
|
|
|
|
valueDisplayUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(keyDislpayUtf16);
|
|
|
|
VBOX_UTF16_FREE(valueDisplayUtf16);
|
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
} else {
|
|
|
|
/* if all are set then default is gui, with vrdp turned on */
|
|
|
|
PRUnichar *keyTypeUtf16 = NULL;
|
|
|
|
PRUnichar *valueTypeUtf16 = NULL;
|
|
|
|
PRUnichar *keyDislpayUtf16 = NULL;
|
|
|
|
PRUnichar *valueDisplayUtf16 = NULL;
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Type", &keyTypeUtf16);
|
|
|
|
VBOX_UTF8_TO_UTF16("gui", &valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetExtraData(machine, keyTypeUtf16, valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(keyTypeUtf16);
|
|
|
|
VBOX_UTF16_FREE(valueTypeUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
if (guiDisplay) {
|
|
|
|
VBOX_UTF8_TO_UTF16("FRONTEND/Display", &keyDislpayUtf16);
|
|
|
|
VBOX_UTF8_TO_UTF16(guiDisplay, &valueDisplayUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
machine->vtbl->SetExtraData(machine, keyDislpayUtf16,
|
|
|
|
valueDisplayUtf16);
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(keyDislpayUtf16);
|
|
|
|
VBOX_UTF16_FREE(valueDisplayUtf16);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
VIR_FREE(guiDisplay);
|
|
|
|
VIR_FREE(sdlDisplay);
|
|
|
|
}
|
2009-05-15 13:45:40 +00:00
|
|
|
|
2010-08-31 18:01:45 +00:00
|
|
|
static void
|
|
|
|
vboxAttachUSB(virDomainDefPtr def, vboxGlobalData *data, IMachine *machine)
|
|
|
|
{
|
|
|
|
IUSBController *USBController = NULL;
|
|
|
|
int i = 0;
|
|
|
|
bool isUSB = false;
|
|
|
|
|
|
|
|
if (def->nhostdevs == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Loop through the devices first and see if you
|
|
|
|
* have a USB Device, only if you have one then
|
|
|
|
* start the USB controller else just proceed as
|
|
|
|
* usual
|
|
|
|
*/
|
|
|
|
for (i = 0; i < def->nhostdevs; i++) {
|
|
|
|
if (def->hostdevs[i]->mode ==
|
|
|
|
VIR_DOMAIN_HOSTDEV_MODE_SUBSYS) {
|
|
|
|
if (def->hostdevs[i]->source.subsys.type ==
|
|
|
|
VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB) {
|
|
|
|
if (def->hostdevs[i]->source.subsys.u.usb.vendor ||
|
|
|
|
def->hostdevs[i]->source.subsys.u.usb.product) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("USB Device detected, VendorId:0x%x, ProductId:0x%x",
|
2010-08-31 18:01:45 +00:00
|
|
|
def->hostdevs[i]->source.subsys.u.usb.vendor,
|
|
|
|
def->hostdevs[i]->source.subsys.u.usb.product);
|
|
|
|
isUSB = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isUSB) {
|
|
|
|
/* First Start the USB Controller and then loop
|
|
|
|
* to attach USB Devices to it
|
|
|
|
*/
|
|
|
|
machine->vtbl->GetUSBController(machine, &USBController);
|
|
|
|
if (USBController) {
|
|
|
|
USBController->vtbl->SetEnabled(USBController, 1);
|
|
|
|
USBController->vtbl->SetEnabledEhci(USBController, 1);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
for (i = 0; i < def->nhostdevs; i++) {
|
|
|
|
if (def->hostdevs[i]->mode ==
|
2010-08-31 18:01:45 +00:00
|
|
|
VIR_DOMAIN_HOSTDEV_MODE_SUBSYS) {
|
2009-12-04 13:49:45 +00:00
|
|
|
if (def->hostdevs[i]->source.subsys.type ==
|
2010-08-31 18:01:45 +00:00
|
|
|
VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB) {
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-08-18 23:31:39 +00:00
|
|
|
char *filtername = NULL;
|
2010-08-31 18:01:45 +00:00
|
|
|
PRUnichar *filternameUtf16 = NULL;
|
|
|
|
IUSBDeviceFilter *filter = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-08-18 23:31:39 +00:00
|
|
|
/* Zero pad for nice alignment when fewer than 9999
|
|
|
|
* devices.
|
2010-08-31 18:01:45 +00:00
|
|
|
*/
|
2010-08-18 23:31:39 +00:00
|
|
|
if (virAsprintf(&filtername, "filter%04d", i) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
} else {
|
|
|
|
VBOX_UTF8_TO_UTF16(filtername, &filternameUtf16);
|
|
|
|
VIR_FREE(filtername);
|
|
|
|
USBController->vtbl->CreateDeviceFilter(USBController,
|
|
|
|
filternameUtf16,
|
|
|
|
&filter);
|
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(filternameUtf16);
|
|
|
|
|
|
|
|
if (filter &&
|
|
|
|
(def->hostdevs[i]->source.subsys.u.usb.vendor ||
|
|
|
|
def->hostdevs[i]->source.subsys.u.usb.product)) {
|
|
|
|
|
|
|
|
PRUnichar *vendorIdUtf16 = NULL;
|
|
|
|
char vendorId[40] = {0};
|
|
|
|
PRUnichar *productIdUtf16 = NULL;
|
|
|
|
char productId[40] = {0};
|
|
|
|
|
|
|
|
if (def->hostdevs[i]->source.subsys.u.usb.vendor) {
|
2010-08-18 23:31:39 +00:00
|
|
|
snprintf(vendorId, sizeof(vendorId), "%x",
|
|
|
|
def->hostdevs[i]->source.subsys.u.usb.vendor);
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(vendorId, &vendorIdUtf16);
|
|
|
|
filter->vtbl->SetVendorId(filter, vendorIdUtf16);
|
|
|
|
VBOX_UTF16_FREE(vendorIdUtf16);
|
|
|
|
}
|
|
|
|
if (def->hostdevs[i]->source.subsys.u.usb.product) {
|
2010-08-18 23:31:39 +00:00
|
|
|
snprintf(productId, sizeof(productId), "%x",
|
|
|
|
def->hostdevs[i]->source.subsys.u.usb.product);
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(productId, &productIdUtf16);
|
|
|
|
filter->vtbl->SetProductId(filter,
|
|
|
|
productIdUtf16);
|
|
|
|
VBOX_UTF16_FREE(productIdUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
filter->vtbl->SetActive(filter, 1);
|
|
|
|
USBController->vtbl->InsertDeviceFilter(USBController,
|
|
|
|
i,
|
|
|
|
filter);
|
|
|
|
VBOX_RELEASE(filter);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_RELEASE(USBController);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainPtr vboxDomainDefineXML(virConnectPtr conn, const char *xml) {
|
|
|
|
VBOX_OBJECT_CHECK(conn, virDomainPtr, NULL);
|
|
|
|
IMachine *machine = NULL;
|
|
|
|
IBIOSSettings *bios = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
|
|
|
vboxIID mchiid = VBOX_IID_INITIALIZER;
|
2010-08-31 18:01:45 +00:00
|
|
|
virDomainDefPtr def = NULL;
|
|
|
|
PRUnichar *machineNameUtf16 = NULL;
|
|
|
|
#if VBOX_API_VERSION >= 3002
|
|
|
|
PRBool override = PR_FALSE;
|
|
|
|
#endif
|
|
|
|
nsresult rc;
|
|
|
|
|
|
|
|
if (!(def = virDomainDefParseString(data->caps, xml,
|
|
|
|
VIR_DOMAIN_XML_INACTIVE))) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16(def->name, &machineNameUtf16);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, def->uuid);
|
2010-08-31 18:01:45 +00:00
|
|
|
#if VBOX_API_VERSION < 3002
|
|
|
|
rc = data->vboxObj->vtbl->CreateMachine(data->vboxObj,
|
|
|
|
machineNameUtf16,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
2010-12-23 16:25:56 +00:00
|
|
|
iid.value,
|
2010-08-31 18:01:45 +00:00
|
|
|
&machine);
|
2010-12-27 22:35:30 +00:00
|
|
|
#elif VBOX_API_VERSION < 4000 /* 3002 <= VBOX_API_VERSION < 4000 */
|
2010-08-31 18:01:45 +00:00
|
|
|
rc = data->vboxObj->vtbl->CreateMachine(data->vboxObj,
|
|
|
|
machineNameUtf16,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
2010-12-23 16:25:56 +00:00
|
|
|
iid.value,
|
2010-08-31 18:01:45 +00:00
|
|
|
override,
|
|
|
|
&machine);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = data->vboxObj->vtbl->CreateMachine(data->vboxObj,
|
|
|
|
NULL,
|
|
|
|
machineNameUtf16,
|
|
|
|
NULL,
|
|
|
|
iid.value,
|
|
|
|
override,
|
|
|
|
&machine);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_UTF16_FREE(machineNameUtf16);
|
|
|
|
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not define a domain, rc=%08x"), (unsigned)rc);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2011-01-28 21:03:24 +00:00
|
|
|
rc = machine->vtbl->SetMemorySize(machine,
|
|
|
|
VIR_DIV_UP(def->mem.cur_balloon, 1024));
|
2010-08-31 18:01:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not set the memory size of the domain to: %lu Kb, "
|
|
|
|
"rc=%08x"),
|
2010-10-12 14:43:39 +00:00
|
|
|
def->mem.cur_balloon, (unsigned)rc);
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
|
|
|
|
2010-09-29 16:20:07 +00:00
|
|
|
if (def->vcpus != def->maxvcpus) {
|
|
|
|
vboxError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
|
|
_("current vcpu count must equal maximum"));
|
|
|
|
}
|
|
|
|
rc = machine->vtbl->SetCPUCount(machine, def->maxvcpus);
|
2010-08-31 18:01:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-09-29 16:20:07 +00:00
|
|
|
_("could not set the number of virtual CPUs to: %u, rc=%08x"),
|
|
|
|
def->maxvcpus, (unsigned)rc);
|
2010-08-31 18:01:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
rc = machine->vtbl->SetPAEEnabled(machine, (def->features) &
|
|
|
|
(1 << VIR_DOMAIN_FEATURE_PAE));
|
|
|
|
#elif VBOX_API_VERSION == 3001
|
|
|
|
rc = machine->vtbl->SetCpuProperty(machine, CpuPropertyType_PAE,
|
|
|
|
(def->features) &
|
|
|
|
(1 << VIR_DOMAIN_FEATURE_PAE));
|
|
|
|
#elif VBOX_API_VERSION >= 3002
|
|
|
|
rc = machine->vtbl->SetCPUProperty(machine, CPUPropertyType_PAE,
|
|
|
|
(def->features) &
|
|
|
|
(1 << VIR_DOMAIN_FEATURE_PAE));
|
|
|
|
#endif
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not change PAE status to: %s, rc=%08x"),
|
|
|
|
((def->features) & (1 << VIR_DOMAIN_FEATURE_PAE))
|
|
|
|
? _("Enabled") : _("Disabled"), (unsigned)rc);
|
|
|
|
}
|
|
|
|
|
|
|
|
machine->vtbl->GetBIOSSettings(machine, &bios);
|
|
|
|
if (bios) {
|
|
|
|
rc = bios->vtbl->SetACPIEnabled(bios, (def->features) &
|
|
|
|
(1 << VIR_DOMAIN_FEATURE_ACPI));
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not change ACPI status to: %s, rc=%08x"),
|
|
|
|
((def->features) & (1 << VIR_DOMAIN_FEATURE_ACPI))
|
|
|
|
? _("Enabled") : _("Disabled"), (unsigned)rc);
|
|
|
|
}
|
|
|
|
rc = bios->vtbl->SetIOAPICEnabled(bios, (def->features) &
|
|
|
|
(1 << VIR_DOMAIN_FEATURE_APIC));
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not change APIC status to: %s, rc=%08x"),
|
|
|
|
((def->features) & (1 << VIR_DOMAIN_FEATURE_APIC))
|
|
|
|
? _("Enabled") : _("Disabled"), (unsigned)rc);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-08-31 18:01:45 +00:00
|
|
|
VBOX_RELEASE(bios);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Register the machine before attaching other devices to it */
|
|
|
|
rc = data->vboxObj->vtbl->RegisterMachine(data->vboxObj, machine);
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not define a domain, rc=%08x"), (unsigned)rc);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the uuid of the machine, currently it is immutable
|
|
|
|
* object so open a session to it and get it back, so that
|
|
|
|
* you can make changes to the machine setting
|
|
|
|
*/
|
2010-12-23 16:25:56 +00:00
|
|
|
machine->vtbl->GetId(machine, &mchiid.value);
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_OPEN(mchiid.value, machine);
|
2010-08-31 18:01:45 +00:00
|
|
|
data->vboxSession->vtbl->GetMachine(data->vboxSession, &machine);
|
|
|
|
|
|
|
|
vboxSetBootDeviceOrder(def, data, machine);
|
|
|
|
vboxAttachDrives(def, data, machine);
|
|
|
|
vboxAttachSound(def, machine);
|
|
|
|
vboxAttachNetwork(def, data, machine);
|
|
|
|
vboxAttachSerial(def, data, machine);
|
|
|
|
vboxAttachParallel(def, data, machine);
|
|
|
|
vboxAttachVideo(def, machine);
|
|
|
|
vboxAttachDisplay(def, data, machine);
|
|
|
|
vboxAttachUSB(def, data, machine);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Save the machine settings made till now and close the
|
|
|
|
* session. also free up the mchiid variable used.
|
|
|
|
*/
|
|
|
|
rc = machine->vtbl->SaveSettings(machine);
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&mchiid);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = virGetDomain(conn, def->name, def->uuid);
|
|
|
|
VBOX_RELEASE(machine);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
virDomainDefFree(def);
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
cleanup:
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
virDomainDefFree(def);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
static int
|
|
|
|
vboxDomainUndefine(virDomainPtr dom)
|
|
|
|
{
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION >= 4000
|
|
|
|
vboxArray media = VBOX_ARRAY_INITIALIZER;
|
|
|
|
#endif
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Block for checking if HDD's are attched to VM.
|
|
|
|
* considering just IDE bus for now. Also skipped
|
|
|
|
* chanel=1 and device=0 (Secondary Master) as currenlty
|
2010-12-27 22:35:30 +00:00
|
|
|
* it is allocated to CD/DVD Drive by default.
|
|
|
|
*
|
|
|
|
* Only do this for VirtualBox 3.x and before. Since
|
|
|
|
* VirtualBox 4.0 the Unregister method can do this for use.
|
2009-12-04 13:49:45 +00:00
|
|
|
*/
|
|
|
|
{
|
|
|
|
PRUnichar *hddcnameUtf16 = NULL;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
char *hddcname = strdup("IDE");
|
|
|
|
VBOX_UTF8_TO_UTF16(hddcname, &hddcnameUtf16);
|
|
|
|
VIR_FREE(hddcname);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Open a Session for the machine */
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
rc = data->vboxSession->vtbl->GetMachine(data->vboxSession, &machine);
|
|
|
|
if (NS_SUCCEEDED(rc) && machine) {
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
# if VBOX_API_VERSION < 3001
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Disconnect all the drives if present */
|
|
|
|
machine->vtbl->DetachHardDisk(machine, hddcnameUtf16, 0, 0);
|
|
|
|
machine->vtbl->DetachHardDisk(machine, hddcnameUtf16, 0, 1);
|
|
|
|
machine->vtbl->DetachHardDisk(machine, hddcnameUtf16, 1, 1);
|
2010-12-27 22:35:30 +00:00
|
|
|
# else /* VBOX_API_VERSION >= 3001 */
|
2009-12-04 13:49:45 +00:00
|
|
|
/* get all the controller first, then the attachments and
|
|
|
|
* remove them all so that the machine can be undefined
|
|
|
|
*/
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray storageControllers = VBOX_ARRAY_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
int i = 0, j = 0;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&storageControllers, machine,
|
|
|
|
machine->vtbl->GetStorageControllers);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < storageControllers.count; i++) {
|
|
|
|
IStorageController *strCtl = storageControllers.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *strCtlName = NULL;
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray mediumAttachments = VBOX_ARRAY_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (!strCtl)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
strCtl->vtbl->GetName(strCtl, &strCtlName);
|
2010-12-27 22:35:30 +00:00
|
|
|
vboxArrayGetWithPtrArg(&mediumAttachments, machine,
|
|
|
|
machine->vtbl->GetMediumAttachmentsOfController,
|
|
|
|
strCtlName);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (j = 0; j < mediumAttachments.count; j++) {
|
|
|
|
IMediumAttachment *medAtt = mediumAttachments.items[j];
|
2009-12-04 13:49:45 +00:00
|
|
|
PRInt32 port = ~0U;
|
|
|
|
PRInt32 device = ~0U;
|
|
|
|
|
|
|
|
if (!medAtt)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
medAtt->vtbl->GetPort(medAtt, &port);
|
|
|
|
medAtt->vtbl->GetDevice(medAtt, &device);
|
|
|
|
|
|
|
|
if ((port != ~0U) && (device != ~0U)) {
|
|
|
|
machine->vtbl->DetachDevice(machine,
|
|
|
|
strCtlName,
|
|
|
|
port,
|
|
|
|
device);
|
|
|
|
}
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&storageControllers);
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->RemoveStorageController(machine, strCtlName);
|
|
|
|
VBOX_UTF16_FREE(strCtlName);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-16 22:05:48 +00:00
|
|
|
|
|
|
|
vboxArrayRelease(&storageControllers);
|
2010-12-27 22:35:30 +00:00
|
|
|
# endif /* VBOX_API_VERSION >= 3001 */
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
machine->vtbl->SaveSettings(machine);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(hddcnameUtf16);
|
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
#endif
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = data->vboxObj->vtbl->UnregisterMachine(data->vboxObj, iid.value, &machine);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
|
|
|
_("no domain with matching uuid"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We're not interested in the array returned by the Unregister method,
|
|
|
|
* but in the side effect of unregistering the virtual machine. In order
|
|
|
|
* to call the Unregister method correctly we need to use the vboxArray
|
|
|
|
* wrapper here. */
|
|
|
|
rc = vboxArrayGetWithUintArg(&media, machine, machine->vtbl->Unregister,
|
|
|
|
CleanupMode_DetachAllReturnNone);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("UUID of machine being undefined", iid.value);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
#if VBOX_API_VERSION < 4000
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->DeleteSettings(machine);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
IProgress *progress = NULL;
|
|
|
|
|
|
|
|
/* The IMachine Delete method takes an array of IMedium items to be
|
|
|
|
* deleted along with the virtual machine. We just want to pass an
|
|
|
|
* empty array. But instead of adding a full vboxArraySetWithReturn to
|
|
|
|
* the glue layer (in order to handle the required signature of the
|
|
|
|
* Delete method) we use a local solution here. */
|
|
|
|
# ifdef WIN32
|
|
|
|
SAFEARRAY *safeArray = NULL;
|
|
|
|
typedef HRESULT __stdcall (*IMachine_Delete)(IMachine *self,
|
|
|
|
SAFEARRAY **media,
|
|
|
|
IProgress **progress);
|
|
|
|
|
|
|
|
((IMachine_Delete)machine->vtbl->Delete)(machine, &safeArray, &progress);
|
|
|
|
# else
|
|
|
|
machine->vtbl->Delete(machine, 0, NULL, &progress);
|
|
|
|
# endif
|
|
|
|
if (progress != NULL) {
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
VBOX_RELEASE(progress);
|
|
|
|
}
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = 0;
|
|
|
|
} else {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not delete the domain, rc=%08x"), (unsigned)rc);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION >= 4000
|
|
|
|
vboxArrayUnalloc(&media);
|
|
|
|
#endif
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-03-22 13:13:53 +00:00
|
|
|
static int vboxDomainAttachDeviceImpl(virDomainPtr dom,
|
|
|
|
const char *xml,
|
|
|
|
int mediaChangeOnly ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state = MachineState_Null;
|
|
|
|
virDomainDefPtr def = NULL;
|
|
|
|
virDomainDeviceDefPtr dev = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
if (VIR_ALLOC(def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
def->os.type = strdup("hvm");
|
|
|
|
|
2009-11-08 21:08:54 +00:00
|
|
|
if (def->os.type == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-11-08 21:08:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-02-09 18:58:01 +00:00
|
|
|
dev = virDomainDeviceDefParse(data->caps, def, xml,
|
2009-04-17 16:09:07 +00:00
|
|
|
VIR_DOMAIN_XML_INACTIVE);
|
|
|
|
if (dev == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching uuid"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (machine) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if ((state == MachineState_Running) ||
|
|
|
|
(state == MachineState_Paused)) {
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN_EXISTING(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
rc = data->vboxSession->vtbl->GetMachine(data->vboxSession, &machine);
|
|
|
|
if (NS_SUCCEEDED(rc) && machine) {
|
|
|
|
if (dev->type == VIR_DOMAIN_DEVICE_DISK) {
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
if (dev->data.disk->device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
2010-03-22 20:01:41 +00:00
|
|
|
if (dev->data.disk->type == VIR_DOMAIN_DISK_TYPE_FILE &&
|
|
|
|
dev->data.disk->src != NULL) {
|
2009-12-04 13:49:45 +00:00
|
|
|
IDVDDrive *dvdDrive = NULL;
|
|
|
|
/* Currently CDROM/DVD Drive is always IDE
|
|
|
|
* Secondary Master so neglecting the following
|
|
|
|
* parameter dev->data.disk->bus
|
|
|
|
*/
|
|
|
|
machine->vtbl->GetDVDDrive(machine, &dvdDrive);
|
|
|
|
if (dvdDrive) {
|
|
|
|
IDVDImage *dvdImage = NULL;
|
|
|
|
PRUnichar *dvdfileUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID dvduuid = VBOX_IID_INITIALIZER;
|
|
|
|
vboxIID dvdemptyuuid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(dev->data.disk->src, &dvdfileUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxObj->vtbl->FindDVDImage(data->vboxObj, dvdfileUtf16, &dvdImage);
|
|
|
|
if (!dvdImage) {
|
2010-12-23 16:25:56 +00:00
|
|
|
data->vboxObj->vtbl->OpenDVDImage(data->vboxObj, dvdfileUtf16, dvdemptyuuid.value, &dvdImage);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
if (dvdImage) {
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = dvdImage->vtbl->imedium.GetId((IMedium *)dvdImage, &dvduuid.value);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't get the uuid of the file to "
|
|
|
|
"be attached to cdrom: %s, rc=%08x"),
|
2009-12-04 13:49:45 +00:00
|
|
|
dev->data.disk->src, (unsigned)rc);
|
|
|
|
} else {
|
|
|
|
/* unmount the previous mounted image */
|
|
|
|
dvdDrive->vtbl->Unmount(dvdDrive);
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = dvdDrive->vtbl->MountImage(dvdDrive, dvduuid.value);
|
2009-04-17 16:09:07 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not attach the file to cdrom: %s, rc=%08x"),
|
2009-04-17 16:09:07 +00:00
|
|
|
dev->data.disk->src, (unsigned)rc);
|
|
|
|
} else {
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = 0;
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("CD/DVD Image UUID:", dvduuid.value);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_MEDIUM_RELEASE(dvdImage);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&dvduuid);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(dvdfileUtf16);
|
|
|
|
VBOX_RELEASE(dvdDrive);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
} else if (dev->data.disk->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
|
|
|
}
|
|
|
|
} else if (dev->data.disk->device == VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
2010-03-22 20:01:41 +00:00
|
|
|
if (dev->data.disk->type == VIR_DOMAIN_DISK_TYPE_FILE &&
|
|
|
|
dev->data.disk->src != NULL) {
|
2009-12-04 13:49:45 +00:00
|
|
|
IFloppyDrive *floppyDrive;
|
|
|
|
machine->vtbl->GetFloppyDrive(machine, &floppyDrive);
|
|
|
|
if (floppyDrive) {
|
|
|
|
rc = floppyDrive->vtbl->SetEnabled(floppyDrive, 1);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
IFloppyImage *floppyImage = NULL;
|
|
|
|
PRUnichar *fdfileUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID fduuid = VBOX_IID_INITIALIZER;
|
|
|
|
vboxIID fdemptyuuid = VBOX_IID_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(dev->data.disk->src, &fdfileUtf16);
|
|
|
|
rc = data->vboxObj->vtbl->FindFloppyImage(data->vboxObj,
|
|
|
|
fdfileUtf16,
|
|
|
|
&floppyImage);
|
|
|
|
|
|
|
|
if (!floppyImage) {
|
|
|
|
data->vboxObj->vtbl->OpenFloppyImage(data->vboxObj,
|
|
|
|
fdfileUtf16,
|
2010-12-23 16:25:56 +00:00
|
|
|
fdemptyuuid.value,
|
2009-12-04 13:49:45 +00:00
|
|
|
&floppyImage);
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (floppyImage) {
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = floppyImage->vtbl->imedium.GetId((IMedium *)floppyImage, &fduuid.value);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("can't get the uuid of the file to be "
|
|
|
|
"attached to floppy drive: %s, rc=%08x"),
|
2009-12-04 13:49:45 +00:00
|
|
|
dev->data.disk->src, (unsigned)rc);
|
|
|
|
} else {
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = floppyDrive->vtbl->MountImage(floppyDrive, fduuid.value);
|
2009-04-17 16:09:07 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not attach the file to floppy drive: %s, rc=%08x"),
|
2009-04-17 16:09:07 +00:00
|
|
|
dev->data.disk->src, (unsigned)rc);
|
|
|
|
} else {
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = 0;
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("attached floppy, UUID:", fduuid.value);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_RELEASE(floppyImage);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&fduuid);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(fdfileUtf16);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(floppyDrive);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
} else if (dev->data.disk->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
|
|
|
} else if (dev->type == VIR_DOMAIN_DEVICE_NET) {
|
|
|
|
} else if (dev->type == VIR_DOMAIN_DEVICE_HOSTDEV) {
|
|
|
|
if (dev->data.hostdev->mode == VIR_DOMAIN_HOSTDEV_MODE_SUBSYS) {
|
|
|
|
if (dev->data.hostdev->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB) {
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->SaveSettings(machine);
|
|
|
|
VBOX_RELEASE(machine);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
virDomainDefFree(def);
|
|
|
|
virDomainDeviceDefFree(dev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-03-22 13:13:53 +00:00
|
|
|
static int vboxDomainAttachDevice(virDomainPtr dom, const char *xml) {
|
|
|
|
return vboxDomainAttachDeviceImpl(dom, xml, 0);
|
|
|
|
}
|
|
|
|
|
2010-01-14 01:44:26 +00:00
|
|
|
static int vboxDomainAttachDeviceFlags(virDomainPtr dom, const char *xml,
|
|
|
|
unsigned int flags) {
|
|
|
|
if (flags & VIR_DOMAIN_DEVICE_MODIFY_CONFIG) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID, "%s",
|
2010-01-14 01:44:26 +00:00
|
|
|
_("cannot modify the persistent configuration of a domain"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-03-22 13:13:53 +00:00
|
|
|
return vboxDomainAttachDeviceImpl(dom, xml, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainUpdateDeviceFlags(virDomainPtr dom, const char *xml,
|
|
|
|
unsigned int flags) {
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(VIR_DOMAIN_DEVICE_MODIFY_CURRENT |
|
|
|
|
VIR_DOMAIN_DEVICE_MODIFY_LIVE |
|
|
|
|
VIR_DOMAIN_DEVICE_MODIFY_CONFIG, -1);
|
|
|
|
|
2010-03-22 13:13:53 +00:00
|
|
|
if (flags & VIR_DOMAIN_DEVICE_MODIFY_CONFIG) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID, "%s",
|
2010-03-22 13:13:53 +00:00
|
|
|
_("cannot modify the persistent configuration of a domain"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return vboxDomainAttachDeviceImpl(dom, xml, 1);
|
2010-01-14 01:44:26 +00:00
|
|
|
}
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
static int vboxDomainDetachDevice(virDomainPtr dom, const char *xml) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2009-04-17 16:09:07 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-04-17 16:09:07 +00:00
|
|
|
PRUint32 state = MachineState_Null;
|
|
|
|
virDomainDefPtr def = NULL;
|
|
|
|
virDomainDeviceDefPtr dev = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-04-17 16:09:07 +00:00
|
|
|
|
|
|
|
if (VIR_ALLOC(def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
def->os.type = strdup("hvm");
|
|
|
|
|
2009-11-08 21:08:54 +00:00
|
|
|
if (def->os.type == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-11-08 21:08:54 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-02-09 18:58:01 +00:00
|
|
|
dev = virDomainDeviceDefParse(data->caps, def, xml,
|
2009-04-17 16:09:07 +00:00
|
|
|
VIR_DOMAIN_XML_INACTIVE);
|
|
|
|
if (dev == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-04-17 16:09:07 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-04-03 19:17:00 +00:00
|
|
|
_("no domain with matching uuid"));
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (machine) {
|
|
|
|
machine->vtbl->GetState(machine, &state);
|
2009-04-17 16:09:07 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if ((state == MachineState_Running) ||
|
|
|
|
(state == MachineState_Paused)) {
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN_EXISTING(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN(iid.value, machine);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
rc = data->vboxSession->vtbl->GetMachine(data->vboxSession, &machine);
|
|
|
|
if (NS_SUCCEEDED(rc) && machine) {
|
|
|
|
if (dev->type == VIR_DOMAIN_DEVICE_DISK) {
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
if (dev->data.disk->device == VIR_DOMAIN_DISK_DEVICE_CDROM) {
|
|
|
|
if (dev->data.disk->type == VIR_DOMAIN_DISK_TYPE_FILE) {
|
|
|
|
IDVDDrive *dvdDrive = NULL;
|
|
|
|
/* Currently CDROM/DVD Drive is always IDE
|
|
|
|
* Secondary Master so neglecting the following
|
|
|
|
* parameter dev->data.disk->bus
|
|
|
|
*/
|
|
|
|
machine->vtbl->GetDVDDrive(machine, &dvdDrive);
|
|
|
|
if (dvdDrive) {
|
|
|
|
rc = dvdDrive->vtbl->Unmount(dvdDrive);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not de-attach the mounted ISO, rc=%08x"),
|
2009-12-04 13:49:45 +00:00
|
|
|
(unsigned)rc);
|
|
|
|
} else {
|
|
|
|
ret = 0;
|
|
|
|
}
|
|
|
|
VBOX_RELEASE(dvdDrive);
|
|
|
|
}
|
|
|
|
} else if (dev->data.disk->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
|
|
|
}
|
|
|
|
} else if (dev->data.disk->device == VIR_DOMAIN_DISK_DEVICE_FLOPPY) {
|
|
|
|
if (dev->data.disk->type == VIR_DOMAIN_DISK_TYPE_FILE) {
|
|
|
|
IFloppyDrive *floppyDrive;
|
|
|
|
machine->vtbl->GetFloppyDrive(machine, &floppyDrive);
|
|
|
|
if (floppyDrive) {
|
|
|
|
PRBool enabled = PR_FALSE;
|
|
|
|
|
|
|
|
floppyDrive->vtbl->GetEnabled(floppyDrive, &enabled);
|
|
|
|
if (enabled) {
|
|
|
|
rc = floppyDrive->vtbl->Unmount(floppyDrive);
|
2009-04-17 16:09:07 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not attach the file "
|
|
|
|
"to floppy drive, rc=%08x"),
|
2009-04-17 16:09:07 +00:00
|
|
|
(unsigned)rc);
|
|
|
|
} else {
|
|
|
|
ret = 0;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
|
|
|
/* If you are here means floppy drive is already unmounted
|
|
|
|
* so don't flag error, just say everything is fine and quit
|
|
|
|
*/
|
|
|
|
ret = 0;
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(floppyDrive);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
} else if (dev->data.disk->type == VIR_DOMAIN_DISK_TYPE_BLOCK) {
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
|
|
|
} else if (dev->type == VIR_DOMAIN_DEVICE_NET) {
|
|
|
|
} else if (dev->type == VIR_DOMAIN_DEVICE_HOSTDEV) {
|
|
|
|
if (dev->data.hostdev->mode == VIR_DOMAIN_HOSTDEV_MODE_SUBSYS) {
|
|
|
|
if (dev->data.hostdev->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB) {
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
machine->vtbl->SaveSettings(machine);
|
|
|
|
VBOX_RELEASE(machine);
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-04-17 16:09:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-04-17 16:09:07 +00:00
|
|
|
virDomainDefFree(def);
|
|
|
|
virDomainDeviceDefFree(dev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-01-14 01:44:26 +00:00
|
|
|
static int vboxDomainDetachDeviceFlags(virDomainPtr dom, const char *xml,
|
|
|
|
unsigned int flags) {
|
|
|
|
if (flags & VIR_DOMAIN_DEVICE_MODIFY_CONFIG) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID, "%s",
|
2010-01-14 01:44:26 +00:00
|
|
|
_("cannot modify the persistent configuration of a domain"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return vboxDomainDetachDevice(dom, xml);
|
|
|
|
}
|
|
|
|
|
2010-03-26 12:40:34 +00:00
|
|
|
static int
|
|
|
|
vboxDomainSnapshotGetAll(virDomainPtr dom,
|
|
|
|
IMachine *machine,
|
|
|
|
ISnapshot ***snapshots)
|
|
|
|
{
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID empty = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
ISnapshot **list = NULL;
|
|
|
|
PRUint32 count;
|
|
|
|
nsresult rc;
|
|
|
|
unsigned int next;
|
|
|
|
unsigned int top;
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetSnapshotCount(machine, &count);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get snapshot count for domain %s"),
|
|
|
|
dom->name);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count == 0)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
if (VIR_ALLOC_N(list, count) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = machine->vtbl->GetSnapshot(machine, empty.value, list);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = machine->vtbl->FindSnapshot(machine, empty.value, list);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc) || !list[0]) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get root snapshot for domain %s"),
|
|
|
|
dom->name);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* BFS walk through snapshot tree */
|
|
|
|
top = 1;
|
|
|
|
for (next = 0; next < count; next++) {
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray children = VBOX_ARRAY_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
if (!list[next]) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("unexpected number of snapshots < %u"), count);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&children, list[next],
|
|
|
|
list[next]->vtbl->GetChildren);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
"%s", _("could not get children snapshots"));
|
|
|
|
goto error;
|
|
|
|
}
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < children.count; i++) {
|
|
|
|
ISnapshot *child = children.items[i];
|
|
|
|
if (!child)
|
2010-03-26 12:40:34 +00:00
|
|
|
continue;
|
|
|
|
if (top == count) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("unexpected number of snapshots > %u"), count);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&children);
|
2010-03-26 12:40:34 +00:00
|
|
|
goto error;
|
|
|
|
}
|
2010-12-16 22:05:48 +00:00
|
|
|
VBOX_ADDREF(child);
|
|
|
|
list[top++] = child;
|
2010-03-26 12:40:34 +00:00
|
|
|
}
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&children);
|
2010-03-26 12:40:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
*snapshots = list;
|
|
|
|
return count;
|
|
|
|
|
|
|
|
error:
|
|
|
|
if (list) {
|
|
|
|
for (next = 0; next < count; next++)
|
|
|
|
VBOX_RELEASE(list[next]);
|
|
|
|
}
|
|
|
|
VIR_FREE(list);
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ISnapshot *
|
|
|
|
vboxDomainSnapshotGet(vboxGlobalData *data,
|
|
|
|
virDomainPtr dom,
|
|
|
|
IMachine *machine,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
ISnapshot **snapshots = NULL;
|
|
|
|
ISnapshot *snapshot = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
int count = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if ((count = vboxDomainSnapshotGetAll(dom, machine, &snapshots)) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
PRUnichar *nameUtf16;
|
|
|
|
char *nameUtf8;
|
|
|
|
|
|
|
|
rc = snapshots[i]->vtbl->GetName(snapshots[i], &nameUtf16);
|
|
|
|
if (NS_FAILED(rc) || !nameUtf16) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
"%s", _("could not get snapshot name"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
VBOX_UTF16_TO_UTF8(nameUtf16, &nameUtf8);
|
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
|
|
|
if (STREQ(name, nameUtf8))
|
|
|
|
snapshot = snapshots[i];
|
|
|
|
VBOX_UTF8_FREE(nameUtf8);
|
|
|
|
|
|
|
|
if (snapshot)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!snapshot) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("domain %s has no snapshots with name %s"),
|
|
|
|
dom->name, name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (count > 0) {
|
|
|
|
for (i = 0; i < count; i++) {
|
|
|
|
if (snapshots[i] != snapshot)
|
|
|
|
VBOX_RELEASE(snapshots[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
VIR_FREE(snapshots);
|
|
|
|
return snapshot;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainSnapshotPtr
|
|
|
|
vboxDomainSnapshotCreateXML(virDomainPtr dom,
|
|
|
|
const char *xmlDesc,
|
2010-04-16 12:04:31 +00:00
|
|
|
unsigned int flags)
|
2010-03-26 12:40:34 +00:00
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, virDomainSnapshotPtr, NULL);
|
|
|
|
virDomainSnapshotDefPtr def = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID domiid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
IConsole *console = NULL;
|
|
|
|
IProgress *progress = NULL;
|
|
|
|
ISnapshot *snapshot = NULL;
|
|
|
|
PRUnichar *name = NULL;
|
|
|
|
PRUnichar *description = NULL;
|
|
|
|
PRUint32 state;
|
|
|
|
nsresult rc;
|
|
|
|
#if VBOX_API_VERSION == 2002
|
|
|
|
nsresult result;
|
|
|
|
#else
|
|
|
|
PRInt32 result;
|
|
|
|
#endif
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(0, NULL);
|
|
|
|
|
2010-03-26 12:40:34 +00:00
|
|
|
if (!(def = virDomainSnapshotDefParseString(xmlDesc, 1)))
|
|
|
|
goto cleanup;
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&domiid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(domiid.value, &machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetState(machine, &state);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get domain state"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((state >= MachineState_FirstOnline)
|
|
|
|
&& (state <= MachineState_LastOnline)) {
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN_EXISTING(domiid.value, machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
} else {
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN(domiid.value, machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_SUCCEEDED(rc))
|
|
|
|
rc = data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not open VirtualBox session with domain %s"),
|
|
|
|
dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16(def->name, &name);
|
|
|
|
if (!name) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (def->description) {
|
|
|
|
VBOX_UTF8_TO_UTF16(def->description, &description);
|
|
|
|
if (!description) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = console->vtbl->TakeSnapshot(console, name, description, &progress);
|
|
|
|
if (NS_FAILED(rc) || !progress) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not take snapshot of domain %s"), dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
progress->vtbl->GetResultCode(progress, &result);
|
|
|
|
if (NS_FAILED(result)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not take snapshot of domain %s"), dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetCurrentSnapshot(machine, &snapshot);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get current snapshot of domain %s"),
|
|
|
|
dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = virGetDomainSnapshot(dom, def->name);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_RELEASE(progress);
|
|
|
|
VBOX_UTF16_FREE(description);
|
|
|
|
VBOX_UTF16_FREE(name);
|
|
|
|
VBOX_RELEASE(console);
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2010-03-26 12:40:34 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&domiid);
|
2010-03-26 12:40:34 +00:00
|
|
|
virDomainSnapshotDefFree(def);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *
|
|
|
|
vboxDomainSnapshotDumpXML(virDomainSnapshotPtr snapshot,
|
2010-04-16 12:04:31 +00:00
|
|
|
unsigned int flags)
|
2010-03-26 12:40:34 +00:00
|
|
|
{
|
|
|
|
virDomainPtr dom = snapshot->domain;
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, char *, NULL);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID domiid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
ISnapshot *snap = NULL;
|
|
|
|
ISnapshot *parent = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
virDomainSnapshotDefPtr def = NULL;
|
|
|
|
PRUnichar *str16;
|
|
|
|
char *str8;
|
|
|
|
PRInt64 timestamp;
|
|
|
|
PRBool online = PR_FALSE;
|
|
|
|
char uuidstr[VIR_UUID_STRING_BUFLEN];
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(0, NULL);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&domiid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(domiid.value, &machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(snap = vboxDomainSnapshotGet(data, dom, machine, snapshot->name)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(def) < 0
|
|
|
|
|| !(def->name = strdup(snapshot->name)))
|
|
|
|
goto no_memory;
|
|
|
|
|
|
|
|
rc = snap->vtbl->GetDescription(snap, &str16);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get description of snapshot %s"),
|
|
|
|
snapshot->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (str16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(str16, &str8);
|
|
|
|
VBOX_UTF16_FREE(str16);
|
|
|
|
def->description = strdup(str8);
|
|
|
|
VBOX_UTF8_FREE(str8);
|
|
|
|
if (!def->description)
|
|
|
|
goto no_memory;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = snap->vtbl->GetTimeStamp(snap, ×tamp);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get creation time of snapshot %s"),
|
|
|
|
snapshot->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
/* timestamp is in milliseconds while creationTime in seconds */
|
|
|
|
def->creationTime = timestamp / 1000;
|
|
|
|
|
|
|
|
rc = snap->vtbl->GetParent(snap, &parent);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get parent of snapshot %s"),
|
|
|
|
snapshot->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (parent) {
|
|
|
|
rc = parent->vtbl->GetName(parent, &str16);
|
|
|
|
if (NS_FAILED(rc) || !str16) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get name of parent of snapshot %s"),
|
|
|
|
snapshot->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
VBOX_UTF16_TO_UTF8(str16, &str8);
|
|
|
|
VBOX_UTF16_FREE(str16);
|
|
|
|
def->parent = strdup(str8);
|
|
|
|
VBOX_UTF8_FREE(str8);
|
|
|
|
if (!def->parent)
|
|
|
|
goto no_memory;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = snap->vtbl->GetOnline(snap, &online);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get online state of snapshot %s"),
|
|
|
|
snapshot->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (online)
|
|
|
|
def->state = VIR_DOMAIN_RUNNING;
|
|
|
|
else
|
|
|
|
def->state = VIR_DOMAIN_SHUTOFF;
|
|
|
|
|
|
|
|
virUUIDFormat(dom->uuid, uuidstr);
|
|
|
|
ret = virDomainSnapshotDefFormat(uuidstr, def, 0);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virDomainSnapshotDefFree(def);
|
|
|
|
VBOX_RELEASE(parent);
|
|
|
|
VBOX_RELEASE(snap);
|
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&domiid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
|
|
|
|
no_memory:
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxDomainSnapshotNum(virDomainPtr dom,
|
2010-04-16 12:04:31 +00:00
|
|
|
unsigned int flags)
|
2010-03-26 12:40:34 +00:00
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
PRUint32 snapshotCount;
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetSnapshotCount(machine, &snapshotCount);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get snapshot count for domain %s"),
|
|
|
|
dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = snapshotCount;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxDomainSnapshotListNames(virDomainPtr dom,
|
|
|
|
char **names,
|
|
|
|
int nameslen,
|
2010-04-16 12:04:31 +00:00
|
|
|
unsigned int flags)
|
2010-03-26 12:40:34 +00:00
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
ISnapshot **snapshots = NULL;
|
|
|
|
int count = 0;
|
|
|
|
int i;
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((count = vboxDomainSnapshotGetAll(dom, machine, &snapshots)) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
for (i = 0; i < nameslen; i++) {
|
|
|
|
PRUnichar *nameUtf16;
|
|
|
|
char *name;
|
|
|
|
|
|
|
|
if (i >= count)
|
|
|
|
break;
|
|
|
|
|
|
|
|
rc = snapshots[i]->vtbl->GetName(snapshots[i], &nameUtf16);
|
|
|
|
if (NS_FAILED(rc) || !nameUtf16) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
"%s", _("could not get snapshot name"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
VBOX_UTF16_TO_UTF8(nameUtf16, &name);
|
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
|
|
|
names[i] = strdup(name);
|
|
|
|
VBOX_UTF8_FREE(name);
|
|
|
|
if (!names[i]) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count <= nameslen)
|
|
|
|
ret = count;
|
|
|
|
else
|
|
|
|
ret = nameslen;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
if (count > 0) {
|
|
|
|
for (i = 0; i < count; i++)
|
|
|
|
VBOX_RELEASE(snapshots[i]);
|
|
|
|
}
|
|
|
|
VIR_FREE(snapshots);
|
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainSnapshotPtr
|
|
|
|
vboxDomainSnapshotLookupByName(virDomainPtr dom,
|
|
|
|
const char *name,
|
2010-04-16 12:04:31 +00:00
|
|
|
unsigned int flags)
|
2010-03-26 12:40:34 +00:00
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, virDomainSnapshotPtr, NULL);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
ISnapshot *snapshot = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(0, NULL);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(snapshot = vboxDomainSnapshotGet(data, dom, machine, name)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = virGetDomainSnapshot(dom, name);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_RELEASE(snapshot);
|
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxDomainHasCurrentSnapshot(virDomainPtr dom,
|
2010-04-16 12:04:31 +00:00
|
|
|
unsigned int flags)
|
2010-03-26 12:40:34 +00:00
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
ISnapshot *snapshot = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetCurrentSnapshot(machine, &snapshot);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get current snapshot"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (snapshot)
|
|
|
|
ret = 1;
|
|
|
|
else
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virDomainSnapshotPtr
|
|
|
|
vboxDomainSnapshotCurrent(virDomainPtr dom,
|
2010-04-16 12:04:31 +00:00
|
|
|
unsigned int flags)
|
2010-03-26 12:40:34 +00:00
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, virDomainSnapshotPtr, NULL);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
ISnapshot *snapshot = NULL;
|
|
|
|
PRUnichar *nameUtf16 = NULL;
|
|
|
|
char *name = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(0, NULL);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(iid.value, &machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetCurrentSnapshot(machine, &snapshot);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get current snapshot"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!snapshot) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("domain has no snapshots"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = snapshot->vtbl->GetName(snapshot, &nameUtf16);
|
|
|
|
if (NS_FAILED(rc) || !nameUtf16) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get current snapshot name"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
VBOX_UTF16_TO_UTF8(nameUtf16, &name);
|
|
|
|
if (!name) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = virGetDomainSnapshot(dom, name);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_UTF8_FREE(name);
|
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
|
|
|
VBOX_RELEASE(snapshot);
|
|
|
|
VBOX_RELEASE(machine);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
static int
|
|
|
|
vboxDomainSnapshotRestore(virDomainPtr dom,
|
|
|
|
IMachine *machine,
|
|
|
|
ISnapshot *snapshot)
|
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
nsresult rc;
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = snapshot->vtbl->GetId(snapshot, &iid.value);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get snapshot UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = machine->vtbl->SetCurrentSnapshot(machine, iid.value);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not restore snapshot for domain %s"), dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static int
|
|
|
|
vboxDomainSnapshotRestore(virDomainPtr dom,
|
|
|
|
IMachine *machine,
|
|
|
|
ISnapshot *snapshot)
|
|
|
|
{
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
|
|
|
IConsole *console = NULL;
|
|
|
|
IProgress *progress = NULL;
|
|
|
|
PRUint32 state;
|
|
|
|
nsresult rc;
|
|
|
|
PRInt32 result;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID domiid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = machine->vtbl->GetId(machine, &domiid.value);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get domain UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetState(machine, &state);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get domain state"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state >= MachineState_FirstOnline
|
|
|
|
&& state <= MachineState_LastOnline) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("domain %s is already running"), dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN(domiid.value, machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_SUCCEEDED(rc))
|
|
|
|
rc = data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not open VirtualBox session with domain %s"),
|
|
|
|
dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = console->vtbl->RestoreSnapshot(console, snapshot, &progress);
|
|
|
|
if (NS_FAILED(rc) || !progress) {
|
|
|
|
if (rc == VBOX_E_INVALID_VM_STATE) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("cannot restore domain snapshot for running domain"));
|
|
|
|
} else {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not restore snapshot for domain %s"),
|
|
|
|
dom->name);
|
|
|
|
}
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
progress->vtbl->GetResultCode(progress, &result);
|
|
|
|
if (NS_FAILED(result)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not restore snapshot for domain %s"), dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_RELEASE(progress);
|
|
|
|
VBOX_RELEASE(console);
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&domiid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxDomainRevertToSnapshot(virDomainSnapshotPtr snapshot,
|
2010-04-16 12:04:31 +00:00
|
|
|
unsigned int flags)
|
2010-03-26 12:40:34 +00:00
|
|
|
{
|
|
|
|
virDomainPtr dom = snapshot->domain;
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID domiid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
ISnapshot *newSnapshot = NULL;
|
|
|
|
ISnapshot *prevSnapshot = NULL;
|
|
|
|
PRBool online = PR_FALSE;
|
|
|
|
PRUint32 state;
|
|
|
|
nsresult rc;
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&domiid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(domiid.value, &machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
newSnapshot = vboxDomainSnapshotGet(data, dom, machine, snapshot->name);
|
|
|
|
if (!newSnapshot)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
rc = newSnapshot->vtbl->GetOnline(newSnapshot, &online);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get online state of snapshot %s"),
|
|
|
|
snapshot->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetCurrentSnapshot(machine, &prevSnapshot);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get current snapshot of domain %s"),
|
|
|
|
dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetState(machine, &state);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get domain state"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state >= MachineState_FirstOnline
|
|
|
|
&& state <= MachineState_LastOnline) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("cannot revert snapshot of running domain"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vboxDomainSnapshotRestore(dom, machine, newSnapshot))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (online) {
|
|
|
|
ret = vboxDomainCreate(dom);
|
|
|
|
if (!ret)
|
|
|
|
vboxDomainSnapshotRestore(dom, machine, prevSnapshot);
|
|
|
|
} else
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_RELEASE(prevSnapshot);
|
|
|
|
VBOX_RELEASE(newSnapshot);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&domiid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxDomainSnapshotDeleteSingle(vboxGlobalData *data,
|
|
|
|
IConsole *console,
|
|
|
|
ISnapshot *snapshot)
|
|
|
|
{
|
|
|
|
IProgress *progress = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
int ret = -1;
|
|
|
|
nsresult rc;
|
|
|
|
#if VBOX_API_VERSION == 2002
|
|
|
|
nsresult result;
|
|
|
|
#else
|
|
|
|
PRInt32 result;
|
|
|
|
#endif
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = snapshot->vtbl->GetId(snapshot, &iid.value);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get snapshot UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = console->vtbl->DiscardSnapshot(console, iid.value, &progress);
|
2010-03-26 12:40:34 +00:00
|
|
|
#else
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = console->vtbl->DeleteSnapshot(console, iid.value, &progress);
|
2010-03-26 12:40:34 +00:00
|
|
|
#endif
|
|
|
|
if (NS_FAILED(rc) || !progress) {
|
|
|
|
if (rc == VBOX_E_INVALID_VM_STATE) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("cannot delete domain snapshot for running domain"));
|
|
|
|
} else {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not delete snapshot"));
|
|
|
|
}
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
progress->vtbl->GetResultCode(progress, &result);
|
|
|
|
if (NS_FAILED(result)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not delete snapshot"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_RELEASE(progress);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxDomainSnapshotDeleteTree(vboxGlobalData *data,
|
|
|
|
IConsole *console,
|
|
|
|
ISnapshot *snapshot)
|
|
|
|
{
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray children = VBOX_ARRAY_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
int ret = -1;
|
|
|
|
nsresult rc;
|
|
|
|
unsigned int i;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&children, snapshot, snapshot->vtbl->GetChildren);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get children snapshots"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < children.count; i++) {
|
|
|
|
if (vboxDomainSnapshotDeleteTree(data, console, children.items[i]))
|
|
|
|
goto cleanup;
|
2010-03-26 12:40:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = vboxDomainSnapshotDeleteSingle(data, console, snapshot);
|
|
|
|
|
|
|
|
cleanup:
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&children);
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
vboxDomainSnapshotDelete(virDomainSnapshotPtr snapshot,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
virDomainPtr dom = snapshot->domain;
|
|
|
|
VBOX_OBJECT_CHECK(dom->conn, int, -1);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID domiid = VBOX_IID_INITIALIZER;
|
2010-03-26 12:40:34 +00:00
|
|
|
IMachine *machine = NULL;
|
|
|
|
ISnapshot *snap = NULL;
|
|
|
|
IConsole *console = NULL;
|
|
|
|
PRUint32 state;
|
|
|
|
nsresult rc;
|
|
|
|
|
2010-04-16 12:04:31 +00:00
|
|
|
virCheckFlags(VIR_DOMAIN_SNAPSHOT_DELETE_CHILDREN, -1);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&domiid, dom->uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(domiid.value, &machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_FAILED(rc)) {
|
2010-12-23 22:21:27 +00:00
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("no domain with matching UUID"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
snap = vboxDomainSnapshotGet(data, dom, machine, snapshot->name);
|
|
|
|
if (!snap)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
rc = machine->vtbl->GetState(machine, &state);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not get domain state"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state >= MachineState_FirstOnline
|
|
|
|
&& state <= MachineState_LastOnline) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_OPERATION_INVALID, "%s",
|
2010-03-26 12:40:34 +00:00
|
|
|
_("cannot delete snapshots of running domain"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
rc = VBOX_SESSION_OPEN(domiid.value, machine);
|
2010-03-26 12:40:34 +00:00
|
|
|
if (NS_SUCCEEDED(rc))
|
|
|
|
rc = data->vboxSession->vtbl->GetConsole(data->vboxSession, &console);
|
|
|
|
if (NS_FAILED(rc)) {
|
2010-04-02 22:38:57 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
2010-03-26 12:40:34 +00:00
|
|
|
_("could not open VirtualBox session with domain %s"),
|
|
|
|
dom->name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & VIR_DOMAIN_SNAPSHOT_DELETE_CHILDREN)
|
|
|
|
ret = vboxDomainSnapshotDeleteTree(data, console, snap);
|
|
|
|
else
|
|
|
|
ret = vboxDomainSnapshotDeleteSingle(data, console, snap);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VBOX_RELEASE(console);
|
|
|
|
VBOX_RELEASE(snap);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&domiid);
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2010-03-26 12:40:34 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION == 2002 || VBOX_API_VERSION == 4000
|
2009-07-24 16:12:16 +00:00
|
|
|
/* No Callback support for VirtualBox 2.2.* series */
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* !(VBOX_API_VERSION == 2002) && !(VBOX_API_VERSION == 4000) */
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
/* Functions needed for Callbacks */
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnMachineStateChange(IVirtualBoxCallback *pThis,
|
|
|
|
PRUnichar *machineId, PRUint32 state)
|
|
|
|
{
|
2009-07-24 16:12:16 +00:00
|
|
|
virDomainPtr dom = NULL;
|
|
|
|
int event = 0;
|
|
|
|
int detail = 0;
|
|
|
|
|
|
|
|
g_pVBoxGlobalData->domainEventDispatching = 1;
|
|
|
|
vboxDriverLock(g_pVBoxGlobalData);
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p, State: %d", pThis, state);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
|
|
|
|
if (machineId) {
|
|
|
|
char *machineIdUtf8 = NULL;
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
|
|
|
|
g_pVBoxGlobalData->pFuncs->pfnUtf16ToUtf8(machineId, &machineIdUtf8);
|
|
|
|
virUUIDParse(machineIdUtf8, uuid);
|
|
|
|
|
|
|
|
dom = vboxDomainLookupByUUID(g_pVBoxGlobalData->conn, uuid);
|
|
|
|
if (dom) {
|
2010-03-18 13:17:14 +00:00
|
|
|
virDomainEventPtr ev;
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
if (state == MachineState_Starting) {
|
|
|
|
event = VIR_DOMAIN_EVENT_STARTED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_STARTED_BOOTED;
|
|
|
|
} else if (state == MachineState_Restoring) {
|
|
|
|
event = VIR_DOMAIN_EVENT_STARTED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_STARTED_RESTORED;
|
|
|
|
} else if (state == MachineState_Paused) {
|
|
|
|
event = VIR_DOMAIN_EVENT_SUSPENDED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_SUSPENDED_PAUSED;
|
|
|
|
} else if (state == MachineState_Running) {
|
|
|
|
event = VIR_DOMAIN_EVENT_RESUMED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_RESUMED_UNPAUSED;
|
|
|
|
} else if (state == MachineState_PoweredOff) {
|
|
|
|
event = VIR_DOMAIN_EVENT_STOPPED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_STOPPED_SHUTDOWN;
|
|
|
|
} else if (state == MachineState_Stopping) {
|
|
|
|
event = VIR_DOMAIN_EVENT_STOPPED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_STOPPED_DESTROYED;
|
|
|
|
} else if (state == MachineState_Aborted) {
|
|
|
|
event = VIR_DOMAIN_EVENT_STOPPED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_STOPPED_CRASHED;
|
|
|
|
} else if (state == MachineState_Saving) {
|
|
|
|
event = VIR_DOMAIN_EVENT_STOPPED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_STOPPED_SAVED;
|
|
|
|
} else {
|
|
|
|
event = VIR_DOMAIN_EVENT_STOPPED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_STOPPED_SHUTDOWN;
|
|
|
|
}
|
|
|
|
|
2010-03-18 13:17:14 +00:00
|
|
|
ev = virDomainEventNewFromDom(dom, event, detail);
|
|
|
|
|
|
|
|
if (ev) {
|
|
|
|
virDomainEventDispatch(ev,
|
|
|
|
g_pVBoxGlobalData->domainEventCallbacks,
|
|
|
|
virDomainEventDispatchDefaultFunc,
|
|
|
|
NULL);
|
|
|
|
virDomainEventFree(ev);
|
2009-07-24 16:12:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainEventCallbackListPurgeMarked(g_pVBoxGlobalData->domainEventCallbacks);
|
|
|
|
|
|
|
|
vboxDriverUnlock(g_pVBoxGlobalData);
|
|
|
|
g_pVBoxGlobalData->domainEventDispatching = 0;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnMachineDataChange(IVirtualBoxCallback *pThis,
|
|
|
|
PRUnichar *machineId)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p", pThis);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnExtraDataCanChange(IVirtualBoxCallback *pThis,
|
|
|
|
PRUnichar *machineId, PRUnichar *key,
|
|
|
|
PRUnichar *value,
|
|
|
|
PRUnichar **error ATTRIBUTE_UNUSED,
|
|
|
|
PRBool *allowChange)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p, allowChange: %s", pThis, *allowChange ? "true" : "false");
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
DEBUGPRUnichar("key", key);
|
|
|
|
DEBUGPRUnichar("value", value);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnExtraDataChange(IVirtualBoxCallback *pThis, PRUnichar *machineId,
|
|
|
|
PRUnichar *key, PRUnichar *value)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p", pThis);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
DEBUGPRUnichar("key", key);
|
|
|
|
DEBUGPRUnichar("value", value);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-03-09 18:22:22 +00:00
|
|
|
# if VBOX_API_VERSION < 3001
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnMediaRegistered(IVirtualBoxCallback *pThis, PRUnichar *mediaId,
|
|
|
|
PRUint32 mediaType, PRBool registered)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p, registered: %s", pThis, registered ? "true" : "false");
|
|
|
|
VIR_DEBUG("mediaType: %d", mediaType);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("mediaId", mediaId);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2010-03-09 18:22:22 +00:00
|
|
|
# else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
# endif /* VBOX_API_VERSION >= 3001 */
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnMachineRegistered(IVirtualBoxCallback *pThis,
|
|
|
|
PRUnichar *machineId, PRBool registered)
|
|
|
|
{
|
2009-07-24 16:12:16 +00:00
|
|
|
virDomainPtr dom = NULL;
|
|
|
|
int event = 0;
|
|
|
|
int detail = 0;
|
|
|
|
|
|
|
|
g_pVBoxGlobalData->domainEventDispatching = 1;
|
|
|
|
vboxDriverLock(g_pVBoxGlobalData);
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p, registered: %s", pThis, registered ? "true" : "false");
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
|
|
|
|
if (machineId) {
|
|
|
|
char *machineIdUtf8 = NULL;
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
|
|
|
|
g_pVBoxGlobalData->pFuncs->pfnUtf16ToUtf8(machineId, &machineIdUtf8);
|
|
|
|
virUUIDParse(machineIdUtf8, uuid);
|
|
|
|
|
|
|
|
dom = vboxDomainLookupByUUID(g_pVBoxGlobalData->conn, uuid);
|
|
|
|
if (dom) {
|
2010-03-18 13:17:14 +00:00
|
|
|
virDomainEventPtr ev;
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
/* CURRENT LIMITATION: we never get the VIR_DOMAIN_EVENT_UNDEFINED
|
|
|
|
* event becuase the when the machine is de-registered the call
|
|
|
|
* to vboxDomainLookupByUUID fails and thus we don't get any
|
|
|
|
* dom pointer which is necessary (null dom pointer doesn't work)
|
|
|
|
* to show the VIR_DOMAIN_EVENT_UNDEFINED event
|
|
|
|
*/
|
|
|
|
if (registered) {
|
|
|
|
event = VIR_DOMAIN_EVENT_DEFINED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_DEFINED_ADDED;
|
|
|
|
} else {
|
|
|
|
event = VIR_DOMAIN_EVENT_UNDEFINED;
|
|
|
|
detail = VIR_DOMAIN_EVENT_UNDEFINED_REMOVED;
|
|
|
|
}
|
|
|
|
|
2010-03-18 13:17:14 +00:00
|
|
|
ev = virDomainEventNewFromDom(dom, event, detail);
|
|
|
|
|
|
|
|
if (ev) {
|
|
|
|
virDomainEventDispatch(ev,
|
|
|
|
g_pVBoxGlobalData->domainEventCallbacks,
|
|
|
|
virDomainEventDispatchDefaultFunc,
|
|
|
|
NULL);
|
|
|
|
virDomainEventFree(ev);
|
2009-07-24 16:12:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virDomainEventCallbackListPurgeMarked(g_pVBoxGlobalData->domainEventCallbacks);
|
|
|
|
|
|
|
|
vboxDriverUnlock(g_pVBoxGlobalData);
|
|
|
|
g_pVBoxGlobalData->domainEventDispatching = 0;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnSessionStateChange(IVirtualBoxCallback *pThis,
|
|
|
|
PRUnichar *machineId, PRUint32 state)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p, state: %d", pThis, state);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnSnapshotTaken(IVirtualBoxCallback *pThis, PRUnichar *machineId,
|
|
|
|
PRUnichar *snapshotId)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p", pThis);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
DEBUGPRUnichar("snapshotId", snapshotId);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnSnapshotDiscarded(IVirtualBoxCallback *pThis, PRUnichar *machineId,
|
|
|
|
PRUnichar *snapshotId)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p", pThis);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
DEBUGPRUnichar("snapshotId", snapshotId);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnSnapshotChange(IVirtualBoxCallback *pThis, PRUnichar *machineId,
|
|
|
|
PRUnichar *snapshotId)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p", pThis);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
DEBUGPRUnichar("snapshotId", snapshotId);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackOnGuestPropertyChange(IVirtualBoxCallback *pThis,
|
|
|
|
PRUnichar *machineId, PRUnichar *name,
|
|
|
|
PRUnichar *value, PRUnichar *flags)
|
|
|
|
{
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("IVirtualBoxCallback: %p", pThis);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGPRUnichar("machineId", machineId);
|
|
|
|
DEBUGPRUnichar("name", name);
|
|
|
|
DEBUGPRUnichar("value", value);
|
|
|
|
DEBUGPRUnichar("flags", flags);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackAddRef(nsISupports *pThis)
|
|
|
|
{
|
2009-07-24 16:12:16 +00:00
|
|
|
nsresult c;
|
|
|
|
|
|
|
|
c = ++g_pVBoxGlobalData->vboxCallBackRefCount;
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("pThis: %p, vboxCallback AddRef: %d", pThis, c);
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackRelease(nsISupports *pThis)
|
|
|
|
{
|
2009-07-24 16:12:16 +00:00
|
|
|
nsresult c;
|
|
|
|
|
|
|
|
c = --g_pVBoxGlobalData->vboxCallBackRefCount;
|
|
|
|
if (c == 0) {
|
|
|
|
/* delete object */
|
|
|
|
VIR_FREE(pThis->vtbl);
|
|
|
|
VIR_FREE(pThis);
|
|
|
|
}
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("pThis: %p, vboxCallback Release: %d", pThis, c);
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2010-12-14 18:53:52 +00:00
|
|
|
static nsresult PR_COM_METHOD
|
|
|
|
vboxCallbackQueryInterface(nsISupports *pThis, const nsID *iid, void **resultp)
|
|
|
|
{
|
2009-07-24 16:12:16 +00:00
|
|
|
IVirtualBoxCallback *that = (IVirtualBoxCallback *)pThis;
|
|
|
|
static const nsID ivirtualboxCallbackUUID = IVIRTUALBOXCALLBACK_IID;
|
|
|
|
static const nsID isupportIID = NS_ISUPPORTS_IID;
|
|
|
|
|
|
|
|
/* Match UUID for IVirtualBoxCallback class */
|
|
|
|
if ( memcmp(iid, &ivirtualboxCallbackUUID, sizeof(nsID)) == 0
|
|
|
|
|| memcmp(iid, &isupportIID, sizeof(nsID)) == 0) {
|
|
|
|
g_pVBoxGlobalData->vboxCallBackRefCount++;
|
|
|
|
*resultp = that;
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("pThis: %p, vboxCallback QueryInterface: %d", pThis, g_pVBoxGlobalData->vboxCallBackRefCount);
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("pThis: %p, vboxCallback QueryInterface didn't find a matching interface", pThis);
|
2009-07-24 16:12:16 +00:00
|
|
|
DEBUGUUID("The UUID Callback Interface expects", iid);
|
|
|
|
DEBUGUUID("The UUID Callback Interface got", &ivirtualboxCallbackUUID);
|
|
|
|
return NS_NOINTERFACE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-04 18:19:08 +00:00
|
|
|
static IVirtualBoxCallback *vboxAllocCallbackObj(void) {
|
2009-07-24 16:12:16 +00:00
|
|
|
IVirtualBoxCallback *vboxCallback = NULL;
|
|
|
|
|
2010-02-15 19:09:55 +00:00
|
|
|
/* Allocate, Initialize and return a valid
|
2009-07-24 16:12:16 +00:00
|
|
|
* IVirtualBoxCallback object here
|
|
|
|
*/
|
|
|
|
if ((VIR_ALLOC(vboxCallback) < 0) || (VIR_ALLOC(vboxCallback->vtbl) < 0)) {
|
2010-02-15 19:09:55 +00:00
|
|
|
VIR_FREE(vboxCallback);
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-07-24 16:12:16 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
vboxCallback->vtbl->nsisupports.AddRef = &vboxCallbackAddRef;
|
|
|
|
vboxCallback->vtbl->nsisupports.Release = &vboxCallbackRelease;
|
|
|
|
vboxCallback->vtbl->nsisupports.QueryInterface = &vboxCallbackQueryInterface;
|
|
|
|
vboxCallback->vtbl->OnMachineStateChange = &vboxCallbackOnMachineStateChange;
|
|
|
|
vboxCallback->vtbl->OnMachineDataChange = &vboxCallbackOnMachineDataChange;
|
|
|
|
vboxCallback->vtbl->OnExtraDataCanChange = &vboxCallbackOnExtraDataCanChange;
|
|
|
|
vboxCallback->vtbl->OnExtraDataChange = &vboxCallbackOnExtraDataChange;
|
2010-03-09 18:22:22 +00:00
|
|
|
# if VBOX_API_VERSION < 3001
|
2009-07-24 16:12:16 +00:00
|
|
|
vboxCallback->vtbl->OnMediaRegistered = &vboxCallbackOnMediaRegistered;
|
2010-03-09 18:22:22 +00:00
|
|
|
# else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
# endif /* VBOX_API_VERSION >= 3001 */
|
2009-07-24 16:12:16 +00:00
|
|
|
vboxCallback->vtbl->OnMachineRegistered = &vboxCallbackOnMachineRegistered;
|
|
|
|
vboxCallback->vtbl->OnSessionStateChange = &vboxCallbackOnSessionStateChange;
|
|
|
|
vboxCallback->vtbl->OnSnapshotTaken = &vboxCallbackOnSnapshotTaken;
|
2010-05-26 11:54:16 +00:00
|
|
|
# if VBOX_API_VERSION < 3002
|
2009-07-24 16:12:16 +00:00
|
|
|
vboxCallback->vtbl->OnSnapshotDiscarded = &vboxCallbackOnSnapshotDiscarded;
|
2010-05-26 11:54:16 +00:00
|
|
|
# else /* VBOX_API_VERSION >= 3002 */
|
|
|
|
vboxCallback->vtbl->OnSnapshotDeleted = &vboxCallbackOnSnapshotDiscarded;
|
|
|
|
# endif /* VBOX_API_VERSION >= 3002 */
|
2009-07-24 16:12:16 +00:00
|
|
|
vboxCallback->vtbl->OnSnapshotChange = &vboxCallbackOnSnapshotChange;
|
|
|
|
vboxCallback->vtbl->OnGuestPropertyChange = &vboxCallbackOnGuestPropertyChange;
|
|
|
|
g_pVBoxGlobalData->vboxCallBackRefCount = 1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return vboxCallback;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void vboxReadCallback(int watch ATTRIBUTE_UNUSED,
|
|
|
|
int fd,
|
|
|
|
int events ATTRIBUTE_UNUSED,
|
|
|
|
void *opaque ATTRIBUTE_UNUSED) {
|
|
|
|
if (fd >= 0) {
|
|
|
|
g_pVBoxGlobalData->vboxQueue->vtbl->ProcessPendingEvents(g_pVBoxGlobalData->vboxQueue);
|
|
|
|
} else {
|
|
|
|
nsresult rc;
|
|
|
|
PLEvent *pEvent = NULL;
|
|
|
|
|
|
|
|
rc = g_pVBoxGlobalData->vboxQueue->vtbl->WaitForEvent(g_pVBoxGlobalData->vboxQueue, &pEvent);
|
|
|
|
if (NS_SUCCEEDED(rc))
|
|
|
|
g_pVBoxGlobalData->vboxQueue->vtbl->HandleEvent(g_pVBoxGlobalData->vboxQueue, pEvent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainEventRegister (virConnectPtr conn,
|
|
|
|
virConnectDomainEventCallback callback,
|
|
|
|
void *opaque,
|
|
|
|
virFreeCallback freecb) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
2009-07-24 16:12:16 +00:00
|
|
|
int vboxRet = -1;
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult rc;
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
/* Locking has to be there as callbacks are not
|
|
|
|
* really fully thread safe
|
|
|
|
*/
|
|
|
|
vboxDriverLock(data);
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (data->vboxCallback == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
data->vboxCallback = vboxAllocCallbackObj();
|
2009-12-04 13:49:45 +00:00
|
|
|
if (data->vboxCallback != NULL) {
|
|
|
|
rc = data->vboxObj->vtbl->RegisterCallback(data->vboxObj, data->vboxCallback);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
vboxRet = 0;
|
2009-07-24 16:12:16 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
|
|
|
vboxRet = 0;
|
|
|
|
}
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Get the vbox file handle and add a event handle to it
|
|
|
|
* so that the events can be passed down to the user
|
|
|
|
*/
|
|
|
|
if (vboxRet == 0) {
|
|
|
|
if (data->fdWatch < 0) {
|
|
|
|
PRInt32 vboxFileHandle;
|
|
|
|
vboxFileHandle = data->vboxQueue->vtbl->GetEventQueueSelectFD(data->vboxQueue);
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
data->fdWatch = virEventAddHandle(vboxFileHandle, VIR_EVENT_HANDLE_READABLE, vboxReadCallback, NULL, NULL);
|
|
|
|
}
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (data->fdWatch >= 0) {
|
|
|
|
/* Once a callback is registered with virtualbox, use a list
|
|
|
|
* to store the callbacks registered with libvirt so that
|
|
|
|
* later you can iterate over them
|
|
|
|
*/
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = virDomainEventCallbackListAdd(conn, data->domainEventCallbacks,
|
|
|
|
callback, opaque, freecb);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("virDomainEventCallbackListAdd (ret = %d) ( conn: %p, "
|
2009-12-04 13:49:45 +00:00
|
|
|
"data->domainEventCallbacks: %p, callback: %p, opaque: %p, "
|
|
|
|
"freecb: %p )", ret, conn, data->domainEventCallbacks, callback,
|
|
|
|
opaque, freecb);
|
2009-07-24 16:12:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vboxDriverUnlock(data);
|
|
|
|
|
|
|
|
if (ret >= 0) {
|
|
|
|
return ret;
|
|
|
|
} else {
|
|
|
|
if (data->vboxObj && data->vboxCallback) {
|
|
|
|
data->vboxObj->vtbl->UnregisterCallback(data->vboxObj, data->vboxCallback);
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainEventDeregister (virConnectPtr conn,
|
|
|
|
virConnectDomainEventCallback callback) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
/* Locking has to be there as callbacks are not
|
|
|
|
* really fully thread safe
|
|
|
|
*/
|
|
|
|
vboxDriverLock(data);
|
|
|
|
|
|
|
|
if (data->domainEventDispatching)
|
|
|
|
ret = virDomainEventCallbackListMarkDelete(conn, data->domainEventCallbacks,
|
|
|
|
callback);
|
|
|
|
else
|
|
|
|
ret = virDomainEventCallbackListRemove(conn, data->domainEventCallbacks,
|
|
|
|
callback);
|
|
|
|
|
2010-03-18 13:17:14 +00:00
|
|
|
if (data->vboxCallback) {
|
2009-07-24 16:12:16 +00:00
|
|
|
/* check count here of how many times register was called
|
|
|
|
* and only on the last de-register do the un-register call
|
|
|
|
*/
|
2010-03-18 13:17:14 +00:00
|
|
|
if (data->domainEventCallbacks && virDomainEventCallbackListCount(data->domainEventCallbacks) == 0) {
|
2009-07-24 16:12:16 +00:00
|
|
|
data->vboxObj->vtbl->UnregisterCallback(data->vboxObj, data->vboxCallback);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(data->vboxCallback);
|
2009-07-24 16:12:16 +00:00
|
|
|
|
|
|
|
/* Remove the Event file handle on which we are listening as well */
|
|
|
|
virEventRemoveHandle(data->fdWatch);
|
|
|
|
data->fdWatch = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vboxDriverUnlock(data);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-03-18 14:47:07 +00:00
|
|
|
static int vboxDomainEventRegisterAny(virConnectPtr conn,
|
|
|
|
virDomainPtr dom,
|
|
|
|
int eventID,
|
|
|
|
virConnectDomainEventGenericCallback callback,
|
|
|
|
void *opaque,
|
|
|
|
virFreeCallback freecb) {
|
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
|
|
|
int vboxRet = -1;
|
|
|
|
nsresult rc;
|
|
|
|
|
|
|
|
/* Locking has to be there as callbacks are not
|
|
|
|
* really fully thread safe
|
|
|
|
*/
|
|
|
|
vboxDriverLock(data);
|
|
|
|
|
|
|
|
if (data->vboxCallback == NULL) {
|
|
|
|
data->vboxCallback = vboxAllocCallbackObj();
|
|
|
|
if (data->vboxCallback != NULL) {
|
|
|
|
rc = data->vboxObj->vtbl->RegisterCallback(data->vboxObj, data->vboxCallback);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
vboxRet = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
vboxRet = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the vbox file handle and add a event handle to it
|
|
|
|
* so that the events can be passed down to the user
|
|
|
|
*/
|
|
|
|
if (vboxRet == 0) {
|
|
|
|
if (data->fdWatch < 0) {
|
|
|
|
PRInt32 vboxFileHandle;
|
|
|
|
vboxFileHandle = data->vboxQueue->vtbl->GetEventQueueSelectFD(data->vboxQueue);
|
|
|
|
|
|
|
|
data->fdWatch = virEventAddHandle(vboxFileHandle, VIR_EVENT_HANDLE_READABLE, vboxReadCallback, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data->fdWatch >= 0) {
|
|
|
|
/* Once a callback is registered with virtualbox, use a list
|
|
|
|
* to store the callbacks registered with libvirt so that
|
|
|
|
* later you can iterate over them
|
|
|
|
*/
|
|
|
|
|
|
|
|
ret = virDomainEventCallbackListAddID(conn, data->domainEventCallbacks,
|
|
|
|
dom, eventID,
|
|
|
|
callback, opaque, freecb);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("virDomainEventCallbackListAddID (ret = %d) ( conn: %p, "
|
2010-03-18 14:47:07 +00:00
|
|
|
"data->domainEventCallbacks: %p, callback: %p, opaque: %p, "
|
|
|
|
"freecb: %p )", ret, conn, data->domainEventCallbacks, callback,
|
|
|
|
opaque, freecb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vboxDriverUnlock(data);
|
|
|
|
|
|
|
|
if (ret >= 0) {
|
|
|
|
return ret;
|
|
|
|
} else {
|
|
|
|
if (data->vboxObj && data->vboxCallback) {
|
|
|
|
data->vboxObj->vtbl->UnregisterCallback(data->vboxObj, data->vboxCallback);
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxDomainEventDeregisterAny(virConnectPtr conn,
|
|
|
|
int callbackID) {
|
|
|
|
VBOX_OBJECT_CHECK(conn, int, -1);
|
|
|
|
|
|
|
|
/* Locking has to be there as callbacks are not
|
|
|
|
* really fully thread safe
|
|
|
|
*/
|
|
|
|
vboxDriverLock(data);
|
|
|
|
|
|
|
|
if (data->domainEventDispatching)
|
|
|
|
ret = virDomainEventCallbackListMarkDeleteID(conn, data->domainEventCallbacks,
|
|
|
|
callbackID);
|
|
|
|
else
|
|
|
|
ret = virDomainEventCallbackListRemoveID(conn, data->domainEventCallbacks,
|
|
|
|
callbackID);
|
|
|
|
|
|
|
|
if (data->vboxCallback) {
|
|
|
|
/* check count here of how many times register was called
|
|
|
|
* and only on the last de-register do the un-register call
|
|
|
|
*/
|
|
|
|
if (data->domainEventCallbacks && virDomainEventCallbackListCount(data->domainEventCallbacks) == 0) {
|
|
|
|
data->vboxObj->vtbl->UnregisterCallback(data->vboxObj, data->vboxCallback);
|
|
|
|
VBOX_RELEASE(data->vboxCallback);
|
|
|
|
|
|
|
|
/* Remove the Event file handle on which we are listening as well */
|
|
|
|
virEventRemoveHandle(data->fdWatch);
|
|
|
|
data->fdWatch = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vboxDriverUnlock(data);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#endif /* !(VBOX_API_VERSION == 2002) && !(VBOX_API_VERSION == 4000) */
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2009-05-15 14:00:50 +00:00
|
|
|
/**
|
|
|
|
* The Network Functions here on
|
|
|
|
*/
|
|
|
|
static virDrvOpenStatus vboxNetworkOpen(virConnectPtr conn,
|
|
|
|
virConnectAuthPtr auth ATTRIBUTE_UNUSED,
|
|
|
|
int flags ATTRIBUTE_UNUSED) {
|
|
|
|
vboxGlobalData *data = conn->privateData;
|
|
|
|
|
|
|
|
if (STRNEQ(conn->driver->name, "VBOX"))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if ((data->pFuncs == NULL) ||
|
|
|
|
(data->vboxObj == NULL) ||
|
|
|
|
(data->vboxSession == NULL))
|
|
|
|
goto cleanup;
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("network initialized");
|
2009-05-15 14:00:50 +00:00
|
|
|
/* conn->networkPrivateData = some network specific data */
|
|
|
|
return VIR_DRV_OPEN_SUCCESS;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
return VIR_DRV_OPEN_DECLINED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxNetworkClose(virConnectPtr conn) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("network uninitialized");
|
2009-05-15 14:00:50 +00:00
|
|
|
conn->networkPrivateData = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxNumOfNetworks(virConnectPtr conn) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(conn, int, 0);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray networkInterfaces = VBOX_ARRAY_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
int i = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&networkInterfaces, host, host->vtbl->GetNetworkInterfaces);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < networkInterfaces.count; i++) {
|
|
|
|
IHostNetworkInterface *networkInterface = networkInterfaces.items[i];
|
|
|
|
|
|
|
|
if (networkInterface) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
PRUint32 status = HostNetworkInterfaceStatus_Unknown;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetStatus(networkInterface, &status);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (status == HostNetworkInterfaceStatus_Up)
|
|
|
|
ret++;
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&networkInterfaces);
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("numActive: %d", ret);
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxListNetworks(virConnectPtr conn, char **const names, int nnames) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(conn, int, 0);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray networkInterfaces = VBOX_ARRAY_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
int i = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&networkInterfaces, host, host->vtbl->GetNetworkInterfaces);
|
|
|
|
|
|
|
|
for (i = 0; (ret < nnames) && (i < networkInterfaces.count); i++) {
|
|
|
|
IHostNetworkInterface *networkInterface = networkInterfaces.items[i];
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
if (networkInterface) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
PRUint32 status = HostNetworkInterfaceStatus_Unknown;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetStatus(networkInterface, &status);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (status == HostNetworkInterfaceStatus_Up) {
|
|
|
|
char *nameUtf8 = NULL;
|
|
|
|
PRUnichar *nameUtf16 = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetName(networkInterface, &nameUtf16);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(nameUtf16, &nameUtf8);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("nnames[%d]: %s", ret, nameUtf8);
|
2009-12-04 13:49:45 +00:00
|
|
|
names[ret] = strdup(nameUtf8);
|
|
|
|
if (names[ret] == NULL) {
|
2010-04-02 22:38:57 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
|
|
|
ret++;
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(nameUtf8);
|
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&networkInterfaces);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(host);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
static int vboxNumOfDefinedNetworks(virConnectPtr conn) {
|
|
|
|
VBOX_OBJECT_HOST_CHECK(conn, int, 0);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray networkInterfaces = VBOX_ARRAY_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
int i = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&networkInterfaces, host, host->vtbl->GetNetworkInterfaces);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < networkInterfaces.count; i++) {
|
|
|
|
IHostNetworkInterface *networkInterface = networkInterfaces.items[i];
|
|
|
|
|
|
|
|
if (networkInterface) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
PRUint32 status = HostNetworkInterfaceStatus_Unknown;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetStatus(networkInterface, &status);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (status == HostNetworkInterfaceStatus_Down)
|
|
|
|
ret++;
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&networkInterfaces);
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("numActive: %d", ret);
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxListDefinedNetworks(virConnectPtr conn, char **const names, int nnames) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(conn, int, 0);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray networkInterfaces = VBOX_ARRAY_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
int i = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&networkInterfaces, host, host->vtbl->GetNetworkInterfaces);
|
|
|
|
|
|
|
|
for (i = 0; (ret < nnames) && (i < networkInterfaces.count); i++) {
|
|
|
|
IHostNetworkInterface *networkInterface = networkInterfaces.items[i];
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
if (networkInterface) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
PRUint32 status = HostNetworkInterfaceStatus_Unknown;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetStatus(networkInterface, &status);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (status == HostNetworkInterfaceStatus_Down) {
|
|
|
|
char *nameUtf8 = NULL;
|
|
|
|
PRUnichar *nameUtf16 = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
networkInterface->vtbl->GetName(networkInterface, &nameUtf16);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(nameUtf16, &nameUtf8);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("nnames[%d]: %s", ret, nameUtf8);
|
2009-12-04 13:49:45 +00:00
|
|
|
names[ret] = strdup(nameUtf8);
|
|
|
|
if (names[ret] == NULL) {
|
2010-04-02 22:38:57 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
|
|
|
ret++;
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(nameUtf8);
|
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&networkInterfaces);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
|
|
|
return ret;
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
static virNetworkPtr
|
|
|
|
vboxNetworkLookupByUUID(virConnectPtr conn, const unsigned char *uuid)
|
|
|
|
{
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(conn, virNetworkPtr, NULL);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-07-24 16:12:16 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&iid, uuid);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-07-24 16:12:16 +00:00
|
|
|
/* TODO: "internal" networks are just strings and
|
|
|
|
* thus can't do much with them
|
|
|
|
*/
|
2009-12-04 13:49:45 +00:00
|
|
|
IHostNetworkInterface *networkInterface = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
host->vtbl->FindHostNetworkInterfaceById(host, iid.value, &networkInterface);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (networkInterface) {
|
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
char *nameUtf8 = NULL;
|
|
|
|
PRUnichar *nameUtf16 = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetName(networkInterface, &nameUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(nameUtf16, &nameUtf8);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = virGetNetwork(conn, nameUtf8, uuid);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Network Name: %s", nameUtf8);
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("Network UUID", iid.value);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(nameUtf8);
|
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-05-15 14:00:50 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virNetworkPtr vboxNetworkLookupByName(virConnectPtr conn, const char *name) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(conn, virNetworkPtr, NULL);
|
|
|
|
PRUnichar *nameUtf16 = NULL;
|
|
|
|
IHostNetworkInterface *networkInterface = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(name, &nameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
host->vtbl->FindHostNetworkInterfaceByName(host, nameUtf16, &networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (networkInterface) {
|
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
networkInterface->vtbl->GetId(networkInterface, &iid.value);
|
|
|
|
vboxIIDToUUID(&iid, uuid);
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = virGetNetwork(conn, name, uuid);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Network Name: %s", name);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("Network UUID", iid.value);
|
|
|
|
vboxIIDUnalloc(&iid);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
2009-05-15 14:00:50 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-09-03 15:06:03 +00:00
|
|
|
static virNetworkPtr vboxNetworkDefineCreateXML(virConnectPtr conn, const char *xml, bool start) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(conn, virNetworkPtr, NULL);
|
|
|
|
PRUnichar *networkInterfaceNameUtf16 = NULL;
|
|
|
|
char *networkInterfaceNameUtf8 = NULL;
|
|
|
|
IHostNetworkInterface *networkInterface = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
nsresult rc;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-02-10 10:22:52 +00:00
|
|
|
virNetworkDefPtr def = virNetworkDefParseString(xml);
|
2010-11-17 18:36:19 +00:00
|
|
|
virNetworkIpDefPtr ipdef;
|
|
|
|
virSocketAddr netmask;
|
2009-09-03 15:06:03 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if ( (!def)
|
2010-11-17 18:36:19 +00:00
|
|
|
|| (def->forwardType != VIR_NETWORK_FORWARD_NONE)
|
|
|
|
|| (def->nips == 0 || !def->ips))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* Look for the first IPv4 IP address definition and use that.
|
|
|
|
* If there weren't any IPv4 addresses, ignore the network (since it's
|
|
|
|
* required below to have an IPv4 address)
|
|
|
|
*/
|
|
|
|
ipdef = virNetworkDefGetIpByIndex(def, AF_INET, 0);
|
|
|
|
if (!ipdef)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (virNetworkIpDefNetmask(ipdef, &netmask) < 0)
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* the current limitation of hostonly network is that you can't
|
|
|
|
* assign a name to it and it defaults to vboxnet*, for e.g:
|
|
|
|
* vboxnet0, vboxnet1, etc. Also the UUID is assigned to it
|
|
|
|
* automatically depending on the mac address and thus both
|
|
|
|
* these paramters are ignored here for now.
|
|
|
|
*/
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-09-03 15:06:03 +00:00
|
|
|
#if VBOX_API_VERSION == 2002
|
2010-12-23 16:25:56 +00:00
|
|
|
if (STREQ(def->name, "vboxnet0")) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(def->name, &networkInterfaceNameUtf16);
|
|
|
|
host->vtbl->FindHostNetworkInterfaceByName(host, networkInterfaceNameUtf16, &networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
|
|
|
if (interfaceType != HostNetworkInterfaceType_HostOnly) {
|
|
|
|
VBOX_RELEASE(networkInterface);
|
|
|
|
networkInterface = NULL;
|
|
|
|
}
|
|
|
|
}
|
2009-09-03 15:06:03 +00:00
|
|
|
#else /* VBOX_API_VERSION != 2002 */
|
2009-12-04 13:49:45 +00:00
|
|
|
IProgress *progress = NULL;
|
|
|
|
host->vtbl->CreateHostOnlyNetworkInterface(host, &networkInterface, &progress);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (progress) {
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
VBOX_RELEASE(progress);
|
|
|
|
}
|
2009-09-03 15:06:03 +00:00
|
|
|
#endif /* VBOX_API_VERSION != 2002 */
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (networkInterface) {
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
char *networkNameUtf8 = NULL;
|
|
|
|
PRUnichar *networkNameUtf16 = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID vboxnetiid = VBOX_IID_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
networkInterface->vtbl->GetName(networkInterface, &networkInterfaceNameUtf16);
|
|
|
|
if (networkInterfaceNameUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(networkInterfaceNameUtf16, &networkInterfaceNameUtf8);
|
|
|
|
|
|
|
|
if (virAsprintf(&networkNameUtf8, "HostInterfaceNetworking-%s", networkInterfaceNameUtf8) < 0) {
|
|
|
|
VBOX_RELEASE(host);
|
|
|
|
VBOX_RELEASE(networkInterface);
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(networkNameUtf8 , &networkNameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Currently support only one dhcp server per network
|
|
|
|
* with contigious address space from start to end
|
|
|
|
*/
|
2010-11-17 18:36:19 +00:00
|
|
|
if ((ipdef->nranges >= 1) &&
|
|
|
|
VIR_SOCKET_HAS_ADDR(&ipdef->ranges[0].start) &&
|
|
|
|
VIR_SOCKET_HAS_ADDR(&ipdef->ranges[0].end)) {
|
2009-12-04 13:49:45 +00:00
|
|
|
IDHCPServer *dhcpServer = NULL;
|
|
|
|
|
|
|
|
data->vboxObj->vtbl->FindDHCPServerByNetworkName(data->vboxObj,
|
|
|
|
networkNameUtf16,
|
|
|
|
&dhcpServer);
|
|
|
|
if (!dhcpServer) {
|
|
|
|
/* create a dhcp server */
|
|
|
|
data->vboxObj->vtbl->CreateDHCPServer(data->vboxObj,
|
|
|
|
networkNameUtf16,
|
|
|
|
&dhcpServer);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("couldn't find dhcp server so creating one");
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
|
|
|
if (dhcpServer) {
|
|
|
|
PRUnichar *ipAddressUtf16 = NULL;
|
|
|
|
PRUnichar *networkMaskUtf16 = NULL;
|
|
|
|
PRUnichar *fromIPAddressUtf16 = NULL;
|
|
|
|
PRUnichar *toIPAddressUtf16 = NULL;
|
|
|
|
PRUnichar *trunkTypeUtf16 = NULL;
|
|
|
|
|
2010-11-17 18:36:19 +00:00
|
|
|
ipAddressUtf16 = vboxSocketFormatAddrUtf16(data, &ipdef->address);
|
|
|
|
networkMaskUtf16 = vboxSocketFormatAddrUtf16(data, &netmask);
|
|
|
|
fromIPAddressUtf16 = vboxSocketFormatAddrUtf16(data, &ipdef->ranges[0].start);
|
|
|
|
toIPAddressUtf16 = vboxSocketFormatAddrUtf16(data, &ipdef->ranges[0].end);
|
2010-10-22 12:21:31 +00:00
|
|
|
|
|
|
|
if (ipAddressUtf16 == NULL || networkMaskUtf16 == NULL ||
|
|
|
|
fromIPAddressUtf16 == NULL || toIPAddressUtf16 == NULL) {
|
|
|
|
VBOX_UTF16_FREE(ipAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(networkMaskUtf16);
|
|
|
|
VBOX_UTF16_FREE(fromIPAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(toIPAddressUtf16);
|
|
|
|
VBOX_RELEASE(dhcpServer);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_UTF8_TO_UTF16("netflt", &trunkTypeUtf16);
|
|
|
|
|
|
|
|
dhcpServer->vtbl->SetEnabled(dhcpServer, PR_TRUE);
|
|
|
|
|
|
|
|
dhcpServer->vtbl->SetConfiguration(dhcpServer,
|
|
|
|
ipAddressUtf16,
|
|
|
|
networkMaskUtf16,
|
|
|
|
fromIPAddressUtf16,
|
|
|
|
toIPAddressUtf16);
|
|
|
|
|
|
|
|
if (start)
|
|
|
|
dhcpServer->vtbl->Start(dhcpServer,
|
|
|
|
networkNameUtf16,
|
|
|
|
networkInterfaceNameUtf16,
|
|
|
|
trunkTypeUtf16);
|
|
|
|
|
|
|
|
VBOX_UTF16_FREE(ipAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(networkMaskUtf16);
|
|
|
|
VBOX_UTF16_FREE(fromIPAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(toIPAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(trunkTypeUtf16);
|
|
|
|
VBOX_RELEASE(dhcpServer);
|
|
|
|
}
|
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-11-17 18:36:19 +00:00
|
|
|
if ((ipdef->nhosts >= 1) &&
|
|
|
|
VIR_SOCKET_HAS_ADDR(&ipdef->hosts[0].ip)) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *ipAddressUtf16 = NULL;
|
|
|
|
PRUnichar *networkMaskUtf16 = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-11-17 18:36:19 +00:00
|
|
|
ipAddressUtf16 = vboxSocketFormatAddrUtf16(data, &ipdef->hosts[0].ip);
|
|
|
|
networkMaskUtf16 = vboxSocketFormatAddrUtf16(data, &netmask);
|
2010-10-22 12:21:31 +00:00
|
|
|
|
|
|
|
if (ipAddressUtf16 == NULL || networkMaskUtf16 == NULL) {
|
|
|
|
VBOX_UTF16_FREE(ipAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(networkMaskUtf16);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* Current drawback is that since EnableStaticIpConfig() sets
|
|
|
|
* IP and enables the interface so even if the dhcpserver is not
|
|
|
|
* started the interface is still up and running
|
|
|
|
*/
|
|
|
|
networkInterface->vtbl->EnableStaticIpConfig(networkInterface,
|
|
|
|
ipAddressUtf16,
|
|
|
|
networkMaskUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(ipAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(networkMaskUtf16);
|
|
|
|
} else {
|
|
|
|
networkInterface->vtbl->EnableDynamicIpConfig(networkInterface);
|
|
|
|
networkInterface->vtbl->DhcpRediscover(networkInterface);
|
|
|
|
}
|
2009-09-03 15:06:03 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = networkInterface->vtbl->GetId(networkInterface, &vboxnetiid.value);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
vboxIIDToUUID(&vboxnetiid, uuid);
|
|
|
|
DEBUGIID("Real Network UUID", vboxnetiid.value);
|
|
|
|
vboxIIDUnalloc(&vboxnetiid);
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = virGetNetwork(conn, networkInterfaceNameUtf8, uuid);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VIR_FREE(networkNameUtf8);
|
|
|
|
VBOX_UTF16_FREE(networkNameUtf16);
|
|
|
|
VBOX_RELEASE(networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(networkInterfaceNameUtf8);
|
|
|
|
VBOX_UTF16_FREE(networkInterfaceNameUtf16);
|
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
2009-05-15 14:00:50 +00:00
|
|
|
cleanup:
|
|
|
|
virNetworkDefFree(def);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-09-03 15:06:03 +00:00
|
|
|
static virNetworkPtr vboxNetworkCreateXML(virConnectPtr conn, const char *xml) {
|
|
|
|
return vboxNetworkDefineCreateXML(conn, xml, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
static virNetworkPtr vboxNetworkDefineXML(virConnectPtr conn, const char *xml) {
|
|
|
|
return vboxNetworkDefineCreateXML(conn, xml, false);
|
|
|
|
}
|
|
|
|
|
2009-09-03 15:08:52 +00:00
|
|
|
static int vboxNetworkUndefineDestroy(virNetworkPtr network, bool removeinterface) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(network->conn, int, -1);
|
2009-05-15 14:00:50 +00:00
|
|
|
char *networkNameUtf8 = NULL;
|
|
|
|
|
|
|
|
/* Current limitation of the function for VirtualBox 2.2.* is
|
|
|
|
* that you can't delete the default hostonly adaptor namely:
|
|
|
|
* vboxnet0 and thus all this functions does is remove the
|
|
|
|
* dhcp server configuration, but the network can still be used
|
|
|
|
* by giving the machine static IP and also it will still
|
|
|
|
* show up in the net-list in virsh
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (virAsprintf(&networkNameUtf8, "HostInterfaceNetworking-%s", network->name) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-05-15 14:00:50 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *networkInterfaceNameUtf16 = NULL;
|
|
|
|
IHostNetworkInterface *networkInterface = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(network->name, &networkInterfaceNameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
host->vtbl->FindHostNetworkInterfaceByName(host, networkInterfaceNameUtf16, &networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (networkInterface) {
|
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
PRUnichar *networkNameUtf16 = NULL;
|
|
|
|
IDHCPServer *dhcpServer = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-09-03 15:08:52 +00:00
|
|
|
#if VBOX_API_VERSION != 2002
|
2009-12-04 13:49:45 +00:00
|
|
|
if (removeinterface) {
|
|
|
|
PRUnichar *iidUtf16 = NULL;
|
|
|
|
IProgress *progress = NULL;
|
2009-09-03 15:08:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetId(networkInterface, &iidUtf16);
|
2009-09-03 15:08:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (iidUtf16) {
|
2010-03-09 18:22:22 +00:00
|
|
|
# if VBOX_API_VERSION == 3000
|
2009-12-04 13:49:45 +00:00
|
|
|
IHostNetworkInterface *netInt = NULL;
|
|
|
|
host->vtbl->RemoveHostOnlyNetworkInterface(host, iidUtf16, &netInt, &progress);
|
|
|
|
VBOX_RELEASE(netInt);
|
2010-03-09 18:22:22 +00:00
|
|
|
# else /* VBOX_API_VERSION > 3000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
host->vtbl->RemoveHostOnlyNetworkInterface(host, iidUtf16, &progress);
|
2010-03-09 18:22:22 +00:00
|
|
|
# endif /* VBOX_API_VERSION > 3000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(iidUtf16);
|
|
|
|
}
|
2009-09-03 15:08:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (progress) {
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
VBOX_RELEASE(progress);
|
|
|
|
}
|
|
|
|
}
|
2009-09-03 15:08:52 +00:00
|
|
|
#endif /* VBOX_API_VERSION != 2002 */
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(networkNameUtf8 , &networkNameUtf16);
|
|
|
|
|
|
|
|
data->vboxObj->vtbl->FindDHCPServerByNetworkName(data->vboxObj,
|
|
|
|
networkNameUtf16,
|
|
|
|
&dhcpServer);
|
|
|
|
if (dhcpServer) {
|
|
|
|
dhcpServer->vtbl->SetEnabled(dhcpServer, PR_FALSE);
|
|
|
|
dhcpServer->vtbl->Stop(dhcpServer);
|
|
|
|
if (removeinterface)
|
|
|
|
data->vboxObj->vtbl->RemoveDHCPServer(data->vboxObj, dhcpServer);
|
|
|
|
VBOX_RELEASE(dhcpServer);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(networkNameUtf16);
|
|
|
|
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(networkInterfaceNameUtf16);
|
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
2009-05-15 14:00:50 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(networkNameUtf8);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-09-03 15:08:52 +00:00
|
|
|
static int vboxNetworkUndefine(virNetworkPtr network) {
|
|
|
|
return vboxNetworkUndefineDestroy(network, true);
|
|
|
|
}
|
|
|
|
|
2009-05-15 14:00:50 +00:00
|
|
|
static int vboxNetworkCreate(virNetworkPtr network) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(network->conn, int, -1);
|
2009-05-15 14:00:50 +00:00
|
|
|
char *networkNameUtf8 = NULL;
|
|
|
|
|
|
|
|
/* Current limitation of the function for VirtualBox 2.2.* is
|
|
|
|
* that the default hostonly network "vboxnet0" is always active
|
|
|
|
* and thus all this functions does is start the dhcp server,
|
|
|
|
* but the network can still be used without starting the dhcp
|
|
|
|
* server by giving the machine static IP
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (virAsprintf(&networkNameUtf8, "HostInterfaceNetworking-%s", network->name) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-05-15 14:00:50 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *networkInterfaceNameUtf16 = NULL;
|
|
|
|
IHostNetworkInterface *networkInterface = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(network->name, &networkInterfaceNameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
host->vtbl->FindHostNetworkInterfaceByName(host, networkInterfaceNameUtf16, &networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (networkInterface) {
|
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
PRUnichar *networkNameUtf16 = NULL;
|
|
|
|
IDHCPServer *dhcpServer = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(networkNameUtf8 , &networkNameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxObj->vtbl->FindDHCPServerByNetworkName(data->vboxObj,
|
|
|
|
networkNameUtf16,
|
|
|
|
&dhcpServer);
|
|
|
|
if (dhcpServer) {
|
|
|
|
PRUnichar *trunkTypeUtf16 = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
dhcpServer->vtbl->SetEnabled(dhcpServer, PR_TRUE);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16("netflt", &trunkTypeUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
dhcpServer->vtbl->Start(dhcpServer,
|
|
|
|
networkNameUtf16,
|
|
|
|
networkInterfaceNameUtf16,
|
|
|
|
trunkTypeUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(trunkTypeUtf16);
|
|
|
|
VBOX_RELEASE(dhcpServer);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(networkNameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(networkInterfaceNameUtf16);
|
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
2009-05-15 14:00:50 +00:00
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(networkNameUtf8);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxNetworkDestroy(virNetworkPtr network) {
|
2009-09-03 15:08:52 +00:00
|
|
|
return vboxNetworkUndefineDestroy(network, false);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static char *vboxNetworkDumpXML(virNetworkPtr network, int flags ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_HOST_CHECK(network->conn, char *, NULL);
|
2009-05-15 14:00:50 +00:00
|
|
|
virNetworkDefPtr def = NULL;
|
2010-11-17 18:36:19 +00:00
|
|
|
virNetworkIpDefPtr ipdef = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
char *networkNameUtf8 = NULL;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(def) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-05-15 14:00:50 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2010-11-17 18:36:19 +00:00
|
|
|
if (VIR_ALLOC(ipdef) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
def->ips = ipdef;
|
|
|
|
def->nips = 1;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
|
|
|
if (virAsprintf(&networkNameUtf8, "HostInterfaceNetworking-%s", network->name) < 0) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-05-15 14:00:50 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *networkInterfaceNameUtf16 = NULL;
|
|
|
|
IHostNetworkInterface *networkInterface = NULL;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(network->name, &networkInterfaceNameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
host->vtbl->FindHostNetworkInterfaceByName(host, networkInterfaceNameUtf16, &networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (networkInterface) {
|
|
|
|
PRUint32 interfaceType = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetInterfaceType(networkInterface, &interfaceType);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (interfaceType == HostNetworkInterfaceType_HostOnly) {
|
|
|
|
def->name = strdup(network->name);
|
|
|
|
if (def->name != NULL) {
|
|
|
|
PRUnichar *networkNameUtf16 = NULL;
|
|
|
|
IDHCPServer *dhcpServer = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID vboxnet0IID = VBOX_IID_INITIALIZER;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
networkInterface->vtbl->GetId(networkInterface, &vboxnet0IID.value);
|
|
|
|
vboxIIDToUUID(&vboxnet0IID, def->uuid);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(networkNameUtf8 , &networkNameUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
def->forwardType = VIR_NETWORK_FORWARD_NONE;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
data->vboxObj->vtbl->FindDHCPServerByNetworkName(data->vboxObj,
|
|
|
|
networkNameUtf16,
|
|
|
|
&dhcpServer);
|
|
|
|
if (dhcpServer) {
|
2010-11-17 18:36:19 +00:00
|
|
|
ipdef->nranges = 1;
|
|
|
|
if (VIR_ALLOC_N(ipdef->ranges, ipdef->nranges) >=0 ) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *ipAddressUtf16 = NULL;
|
|
|
|
PRUnichar *networkMaskUtf16 = NULL;
|
|
|
|
PRUnichar *fromIPAddressUtf16 = NULL;
|
|
|
|
PRUnichar *toIPAddressUtf16 = NULL;
|
2010-10-22 12:21:31 +00:00
|
|
|
bool errorOccurred = false;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
dhcpServer->vtbl->GetIPAddress(dhcpServer, &ipAddressUtf16);
|
|
|
|
dhcpServer->vtbl->GetNetworkMask(dhcpServer, &networkMaskUtf16);
|
|
|
|
dhcpServer->vtbl->GetLowerIP(dhcpServer, &fromIPAddressUtf16);
|
|
|
|
dhcpServer->vtbl->GetUpperIP(dhcpServer, &toIPAddressUtf16);
|
|
|
|
/* Currently virtualbox supports only one dhcp server per network
|
|
|
|
* with contigious address space from start to end
|
|
|
|
*/
|
2010-10-22 12:21:31 +00:00
|
|
|
if (vboxSocketParseAddrUtf16(data, ipAddressUtf16,
|
2010-11-17 18:36:19 +00:00
|
|
|
&ipdef->address) < 0 ||
|
2010-10-22 12:21:31 +00:00
|
|
|
vboxSocketParseAddrUtf16(data, networkMaskUtf16,
|
2010-11-17 18:36:19 +00:00
|
|
|
&ipdef->netmask) < 0 ||
|
2010-10-22 12:21:31 +00:00
|
|
|
vboxSocketParseAddrUtf16(data, fromIPAddressUtf16,
|
2010-11-17 18:36:19 +00:00
|
|
|
&ipdef->ranges[0].start) < 0 ||
|
2010-10-22 12:21:31 +00:00
|
|
|
vboxSocketParseAddrUtf16(data, toIPAddressUtf16,
|
2010-11-17 18:36:19 +00:00
|
|
|
&ipdef->ranges[0].end) < 0) {
|
2010-10-22 12:21:31 +00:00
|
|
|
errorOccurred = true;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_UTF16_FREE(ipAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(networkMaskUtf16);
|
|
|
|
VBOX_UTF16_FREE(fromIPAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(toIPAddressUtf16);
|
2010-10-22 12:21:31 +00:00
|
|
|
|
|
|
|
if (errorOccurred) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-11-17 18:36:19 +00:00
|
|
|
ipdef->nranges = 0;
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2010-11-17 18:36:19 +00:00
|
|
|
ipdef->nhosts = 1;
|
|
|
|
if (VIR_ALLOC_N(ipdef->hosts, ipdef->nhosts) >=0 ) {
|
|
|
|
ipdef->hosts[0].name = strdup(network->name);
|
|
|
|
if (ipdef->hosts[0].name == NULL) {
|
|
|
|
VIR_FREE(ipdef->hosts);
|
|
|
|
ipdef->nhosts = 0;
|
2010-04-02 22:38:57 +00:00
|
|
|
virReportOOMError();
|
2009-05-15 14:00:50 +00:00
|
|
|
} else {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *macAddressUtf16 = NULL;
|
|
|
|
PRUnichar *ipAddressUtf16 = NULL;
|
2010-10-22 12:21:31 +00:00
|
|
|
bool errorOccurred = false;
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
networkInterface->vtbl->GetHardwareAddress(networkInterface, &macAddressUtf16);
|
2009-05-15 14:00:50 +00:00
|
|
|
networkInterface->vtbl->GetIPAddress(networkInterface, &ipAddressUtf16);
|
|
|
|
|
2010-11-17 18:36:19 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(macAddressUtf16, &ipdef->hosts[0].mac);
|
2010-10-22 12:21:31 +00:00
|
|
|
|
|
|
|
if (vboxSocketParseAddrUtf16(data, ipAddressUtf16,
|
2010-11-17 18:36:19 +00:00
|
|
|
&ipdef->hosts[0].ip) < 0) {
|
2010-10-22 12:21:31 +00:00
|
|
|
errorOccurred = true;
|
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(macAddressUtf16);
|
|
|
|
VBOX_UTF16_FREE(ipAddressUtf16);
|
2010-10-22 12:21:31 +00:00
|
|
|
|
|
|
|
if (errorOccurred) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
} else {
|
2010-11-17 18:36:19 +00:00
|
|
|
ipdef->nhosts = 0;
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(dhcpServer);
|
|
|
|
} else {
|
|
|
|
PRUnichar *networkMaskUtf16 = NULL;
|
|
|
|
PRUnichar *ipAddressUtf16 = NULL;
|
2010-10-22 12:21:31 +00:00
|
|
|
bool errorOccurred = false;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
networkInterface->vtbl->GetNetworkMask(networkInterface, &networkMaskUtf16);
|
|
|
|
networkInterface->vtbl->GetIPAddress(networkInterface, &ipAddressUtf16);
|
|
|
|
|
2010-10-22 12:21:31 +00:00
|
|
|
if (vboxSocketParseAddrUtf16(data, networkMaskUtf16,
|
2010-11-17 18:36:19 +00:00
|
|
|
&ipdef->netmask) < 0 ||
|
2010-10-22 12:21:31 +00:00
|
|
|
vboxSocketParseAddrUtf16(data, ipAddressUtf16,
|
2010-11-17 18:36:19 +00:00
|
|
|
&ipdef->address) < 0) {
|
2010-10-22 12:21:31 +00:00
|
|
|
errorOccurred = true;
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_UTF16_FREE(networkMaskUtf16);
|
|
|
|
VBOX_UTF16_FREE(ipAddressUtf16);
|
2010-10-22 12:21:31 +00:00
|
|
|
|
|
|
|
if (errorOccurred) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("Network UUID", vboxnet0IID.value);
|
|
|
|
vboxIIDUnalloc(&vboxnet0IID);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(networkNameUtf16);
|
|
|
|
} else {
|
2010-04-02 22:38:57 +00:00
|
|
|
virReportOOMError();
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_RELEASE(networkInterface);
|
2009-05-15 14:00:50 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(networkInterfaceNameUtf16);
|
|
|
|
VBOX_RELEASE(host);
|
|
|
|
|
2010-02-10 10:22:52 +00:00
|
|
|
ret = virNetworkDefFormat(def);
|
2009-05-15 14:00:50 +00:00
|
|
|
|
|
|
|
cleanup:
|
2010-02-15 16:54:15 +00:00
|
|
|
virNetworkDefFree(def);
|
2009-05-15 14:00:50 +00:00
|
|
|
VIR_FREE(networkNameUtf8);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2009-09-04 14:28:52 +00:00
|
|
|
/**
|
|
|
|
* The Storage Functions here on
|
|
|
|
*/
|
|
|
|
|
|
|
|
static virDrvOpenStatus vboxStorageOpen (virConnectPtr conn,
|
|
|
|
virConnectAuthPtr auth ATTRIBUTE_UNUSED,
|
|
|
|
int flags ATTRIBUTE_UNUSED) {
|
|
|
|
vboxGlobalData *data = conn->privateData;
|
|
|
|
|
|
|
|
if (STRNEQ(conn->driver->name, "VBOX"))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if ((data->pFuncs == NULL) ||
|
|
|
|
(data->vboxObj == NULL) ||
|
|
|
|
(data->vboxSession == NULL))
|
|
|
|
goto cleanup;
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("vbox storage initialized");
|
2009-09-04 14:28:52 +00:00
|
|
|
/* conn->storagePrivateData = some storage specific data */
|
|
|
|
return VIR_DRV_OPEN_SUCCESS;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
return VIR_DRV_OPEN_DECLINED;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxStorageClose (virConnectPtr conn) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("vbox storage uninitialized");
|
2009-09-04 14:28:52 +00:00
|
|
|
conn->storagePrivateData = NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxStorageNumOfPools(virConnectPtr conn ATTRIBUTE_UNUSED) {
|
|
|
|
|
|
|
|
/** Currently only one pool supported, the default one
|
|
|
|
* given by ISystemProperties::defaultHardDiskFolder()
|
|
|
|
*/
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2010-02-04 18:19:08 +00:00
|
|
|
static int vboxStorageListPools(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|
|
|
char **const names, int nnames) {
|
2009-09-04 14:28:52 +00:00
|
|
|
int numActive = 0;
|
|
|
|
|
|
|
|
if (nnames == 1) {
|
|
|
|
names[numActive] = strdup("default-pool");
|
|
|
|
if (names[numActive] == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-09-04 14:28:52 +00:00
|
|
|
} else {
|
|
|
|
numActive++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return numActive;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virStoragePoolPtr vboxStoragePoolLookupByName(virConnectPtr conn, const char *name) {
|
|
|
|
virStoragePoolPtr ret = NULL;
|
|
|
|
|
|
|
|
/** Current limitation of the function: since
|
|
|
|
* the default pool doesn't have UUID just assign
|
|
|
|
* one till vbox can handle pools
|
|
|
|
*/
|
|
|
|
if (STREQ("default-pool", name)) {
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
const char *uuidstr = "1deff1ff-1481-464f-967f-a50fe8936cc4";
|
|
|
|
|
|
|
|
virUUIDParse(uuidstr, uuid);
|
|
|
|
|
|
|
|
ret = virGetStoragePool(conn, name, uuid);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxStoragePoolNumOfVolumes(virStoragePoolPtr pool) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(pool->conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray hardDisks = VBOX_ARRAY_INITIALIZER;
|
2009-09-04 14:28:52 +00:00
|
|
|
PRUint32 hardDiskAccessible = 0;
|
|
|
|
nsresult rc;
|
|
|
|
int i;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&hardDisks, data->vboxObj, data->vboxObj->vtbl->GetHardDisks);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < hardDisks.count; ++i) {
|
|
|
|
IHardDisk *hardDisk = hardDisks.items[i];
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hardDisk) {
|
|
|
|
PRUint32 hddstate;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (hddstate != MediaState_Inaccessible)
|
|
|
|
hardDiskAccessible++;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
}
|
2010-12-16 22:05:48 +00:00
|
|
|
|
|
|
|
vboxArrayRelease(&hardDisks);
|
|
|
|
|
|
|
|
ret = hardDiskAccessible;
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-12-16 22:05:48 +00:00
|
|
|
ret = -1;
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not get number of volumes in the pool: %s, rc=%08x"),
|
2009-12-04 13:49:45 +00:00
|
|
|
pool->name, (unsigned)rc);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxStoragePoolListVolumes(virStoragePoolPtr pool, char **const names, int nnames) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(pool->conn, int, -1);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray hardDisks = VBOX_ARRAY_INITIALIZER;
|
2009-09-04 14:28:52 +00:00
|
|
|
PRUint32 numActive = 0;
|
|
|
|
nsresult rc;
|
|
|
|
int i;
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&hardDisks, data->vboxObj, data->vboxObj->vtbl->GetHardDisks);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < hardDisks.count && numActive < nnames; ++i) {
|
|
|
|
IHardDisk *hardDisk = hardDisks.items[i];
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hardDisk) {
|
|
|
|
PRUint32 hddstate;
|
|
|
|
char *nameUtf8 = NULL;
|
|
|
|
PRUnichar *nameUtf16 = NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (hddstate != MediaState_Inaccessible) {
|
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetName, &nameUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(nameUtf16, &nameUtf8);
|
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (nameUtf8) {
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("nnames[%d]: %s", numActive, nameUtf8);
|
2009-12-04 13:49:45 +00:00
|
|
|
names[numActive] = strdup(nameUtf8);
|
|
|
|
if (names[numActive] == NULL) {
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
|
|
|
numActive++;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_UTF8_FREE(nameUtf8);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-12-16 22:05:48 +00:00
|
|
|
|
|
|
|
vboxArrayRelease(&hardDisks);
|
|
|
|
|
|
|
|
ret = numActive;
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
2010-12-16 22:05:48 +00:00
|
|
|
ret = -1;
|
2010-04-03 19:17:00 +00:00
|
|
|
vboxError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("could not get the volume list in the pool: %s, rc=%08x"),
|
2009-12-04 13:49:45 +00:00
|
|
|
pool->name, (unsigned)rc);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static virStorageVolPtr vboxStorageVolLookupByName(virStoragePoolPtr pool, const char *name) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(pool->conn, virStorageVolPtr, NULL);
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray hardDisks = VBOX_ARRAY_INITIALIZER;
|
2009-09-04 14:28:52 +00:00
|
|
|
nsresult rc;
|
|
|
|
int i;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if(!name)
|
|
|
|
return ret;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
rc = vboxArrayGet(&hardDisks, data->vboxObj, data->vboxObj->vtbl->GetHardDisks);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < hardDisks.count; ++i) {
|
|
|
|
IHardDisk *hardDisk = hardDisks.items[i];
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hardDisk) {
|
|
|
|
PRUint32 hddstate;
|
|
|
|
char *nameUtf8 = NULL;
|
|
|
|
PRUnichar *nameUtf16 = NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (hddstate != MediaState_Inaccessible) {
|
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetName, &nameUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (nameUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(nameUtf16, &nameUtf8);
|
|
|
|
VBOX_UTF16_FREE(nameUtf16);
|
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (nameUtf8 && STREQ(nameUtf8, name)) {
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID hddIID = VBOX_IID_INITIALIZER;
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
char key[VIR_UUID_STRING_BUFLEN] = "";
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetId, &hddIID.value);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
vboxIIDToUUID(&hddIID, uuid);
|
|
|
|
virUUIDFormat(uuid, key);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
ret = virGetStorageVol(pool->conn, pool->name, name, key);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("virStorageVolPtr: %p", ret);
|
|
|
|
VIR_DEBUG("Storage Volume Name: %s", name);
|
|
|
|
VIR_DEBUG("Storage Volume key : %s", key);
|
|
|
|
VIR_DEBUG("Storage Volume Pool: %s", pool->name);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&hddIID);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(nameUtf8);
|
|
|
|
break;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (nameUtf8)
|
|
|
|
VBOX_UTF8_FREE(nameUtf8);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&hardDisks);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virStorageVolPtr vboxStorageVolLookupByKey(virConnectPtr conn, const char *key) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, virStorageVolPtr, NULL);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID hddIID = VBOX_IID_INITIALIZER;
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
2009-09-04 14:28:52 +00:00
|
|
|
IHardDisk *hardDisk = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!key)
|
|
|
|
return ret;
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (virUUIDParse(key, uuid) < 0) {
|
|
|
|
vboxError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("Could not parse UUID from '%s'"), key);
|
|
|
|
return NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&hddIID, uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = data->vboxObj->vtbl->GetHardDisk(data->vboxObj, hddIID.value, &hardDisk);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = data->vboxObj->vtbl->FindMedium(data->vboxObj, hddIID.value,
|
|
|
|
DeviceType_HardDisk, &hardDisk);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
PRUint32 hddstate;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (hddstate != MediaState_Inaccessible) {
|
|
|
|
PRUnichar *hddNameUtf16 = NULL;
|
|
|
|
char *hddNameUtf8 = NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetName, &hddNameUtf16);
|
|
|
|
VBOX_UTF16_TO_UTF8(hddNameUtf16, &hddNameUtf8);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hddNameUtf8) {
|
|
|
|
if (vboxStorageNumOfPools(conn) == 1) {
|
|
|
|
ret = virGetStorageVol(conn, "default-pool", hddNameUtf8, key);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Storage Volume Pool: %s", "default-pool");
|
2009-12-04 13:49:45 +00:00
|
|
|
} else {
|
|
|
|
/* TODO: currently only one default pool and thus
|
|
|
|
* nothing here, change it when pools are supported
|
|
|
|
*/
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Storage Volume Name: %s", key);
|
|
|
|
VIR_DEBUG("Storage Volume key : %s", hddNameUtf8);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_UTF8_FREE(hddNameUtf8);
|
|
|
|
VBOX_UTF16_FREE(hddNameUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDisk);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&hddIID);
|
2009-09-04 14:28:52 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virStorageVolPtr vboxStorageVolLookupByPath(virConnectPtr conn, const char *path) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(conn, virStorageVolPtr, NULL);
|
2009-09-04 14:28:52 +00:00
|
|
|
PRUnichar *hddPathUtf16 = NULL;
|
|
|
|
IHardDisk *hardDisk = NULL;
|
|
|
|
nsresult rc;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!path)
|
|
|
|
return ret;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(path, &hddPathUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (!hddPathUtf16)
|
|
|
|
return ret;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2009-12-04 13:49:45 +00:00
|
|
|
rc = data->vboxObj->vtbl->FindHardDisk(data->vboxObj, hddPathUtf16, &hardDisk);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = data->vboxObj->vtbl->FindMedium(data->vboxObj, hddPathUtf16,
|
|
|
|
DeviceType_HardDisk, &hardDisk);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
PRUint32 hddstate;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (hddstate != MediaState_Inaccessible) {
|
|
|
|
PRUnichar *hddNameUtf16 = NULL;
|
|
|
|
char *hddNameUtf8 = NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetName, &hddNameUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hddNameUtf16) {
|
|
|
|
VBOX_UTF16_TO_UTF8(hddNameUtf16, &hddNameUtf8);
|
|
|
|
VBOX_UTF16_FREE(hddNameUtf16);
|
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (hddNameUtf8) {
|
|
|
|
vboxIID hddIID = VBOX_IID_INITIALIZER;
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
char key[VIR_UUID_STRING_BUFLEN] = "";
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetId, &hddIID.value);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
vboxIIDToUUID(&hddIID, uuid);
|
|
|
|
virUUIDFormat(uuid, key);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
/* TODO: currently only one default pool and thus
|
|
|
|
* the check below, change it when pools are supported
|
|
|
|
*/
|
|
|
|
if (vboxStorageNumOfPools(conn) == 1)
|
|
|
|
ret = virGetStorageVol(conn, "default-pool", hddNameUtf8, key);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Storage Volume Pool: %s", "default-pool");
|
|
|
|
VIR_DEBUG("Storage Volume Name: %s", hddNameUtf8);
|
|
|
|
VIR_DEBUG("Storage Volume key : %s", key);
|
2010-12-23 16:25:56 +00:00
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&hddIID);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hddNameUtf8)
|
|
|
|
VBOX_UTF8_FREE(hddNameUtf8);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDisk);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(hddPathUtf16);
|
|
|
|
|
2009-09-04 14:28:52 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static virStorageVolPtr vboxStorageVolCreateXML(virStoragePoolPtr pool,
|
|
|
|
const char *xml,
|
|
|
|
unsigned int flags ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(pool->conn, virStorageVolPtr, NULL);
|
2009-09-04 14:28:52 +00:00
|
|
|
virStorageVolDefPtr def = NULL;
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *hddFormatUtf16 = NULL;
|
|
|
|
PRUnichar *hddNameUtf16 = NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
virStoragePoolDef poolDef;
|
|
|
|
nsresult rc;
|
|
|
|
|
|
|
|
/* since there is currently one default pool now
|
|
|
|
* and virStorageVolDefFormat() just checks it type
|
|
|
|
* so just assign it for now, change the behaviour
|
|
|
|
* when vbox supports pools.
|
|
|
|
*/
|
|
|
|
memset(&poolDef, 0, sizeof(poolDef));
|
|
|
|
poolDef.type = VIR_STORAGE_POOL_DIR;
|
|
|
|
|
2010-02-10 11:42:56 +00:00
|
|
|
if ((def = virStorageVolDefParseString(&poolDef, xml)) == NULL)
|
2009-09-04 14:28:52 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if ( !def->name
|
|
|
|
|| (def->type != VIR_STORAGE_VOL_FILE))
|
|
|
|
goto cleanup;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* TODO: for now only the vmdk, vpc and vdi type harddisk
|
|
|
|
* variants can be created, also since there is no vdi
|
|
|
|
* type in enum virStorageFileFormat {} the default
|
|
|
|
* will be to create vdi if nothing is specified in
|
|
|
|
* def->target.format
|
|
|
|
*/
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (def->target.format == VIR_STORAGE_FILE_VMDK) {
|
|
|
|
VBOX_UTF8_TO_UTF16("VMDK", &hddFormatUtf16);
|
|
|
|
} else if (def->target.format == VIR_STORAGE_FILE_VPC) {
|
|
|
|
VBOX_UTF8_TO_UTF16("VHD", &hddFormatUtf16);
|
|
|
|
} else {
|
|
|
|
VBOX_UTF8_TO_UTF16("VDI", &hddFormatUtf16);
|
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_TO_UTF16(def->name, &hddNameUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (hddFormatUtf16 && hddNameUtf16) {
|
|
|
|
IHardDisk *hardDisk = NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
rc = data->vboxObj->vtbl->CreateHardDisk(data->vboxObj, hddFormatUtf16, hddNameUtf16, &hardDisk);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
IProgress *progress = NULL;
|
2011-01-28 21:03:24 +00:00
|
|
|
PRUint64 logicalSize = VIR_DIV_UP(def->capacity, 1024 * 1024);
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint32 variant = HardDiskVariant_Standard;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (def->capacity == def->allocation)
|
|
|
|
variant = HardDiskVariant_Fixed;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
rc = hardDisk->vtbl->CreateBaseStorage(hardDisk, logicalSize, variant, &progress);
|
|
|
|
if (NS_SUCCEEDED(rc) && progress) {
|
2009-09-04 14:28:52 +00:00
|
|
|
#if VBOX_API_VERSION == 2002
|
2009-12-04 13:49:45 +00:00
|
|
|
nsresult resultCode;
|
2009-09-04 14:28:52 +00:00
|
|
|
#else
|
2009-12-04 13:49:45 +00:00
|
|
|
PRInt32 resultCode;
|
2009-09-04 14:28:52 +00:00
|
|
|
#endif
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
progress->vtbl->GetResultCode(progress, &resultCode);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(resultCode)) {
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID hddIID = VBOX_IID_INITIALIZER;
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
char key[VIR_UUID_STRING_BUFLEN] = "";
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetId, &hddIID.value);
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDToUUID(&hddIID, uuid);
|
|
|
|
virUUIDFormat(uuid, key);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
ret = virGetStorageVol(pool->conn, pool->name, def->name, key);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
|
|
|
|
vboxIIDUnalloc(&hddIID);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(progress);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(hddFormatUtf16);
|
|
|
|
VBOX_UTF16_FREE(hddNameUtf16);
|
|
|
|
|
2009-09-04 14:28:52 +00:00
|
|
|
cleanup:
|
|
|
|
virStorageVolDefFree(def);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxStorageVolDelete(virStorageVolPtr vol,
|
|
|
|
unsigned int flags ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(vol->conn, int, -1);
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID hddIID = VBOX_IID_INITIALIZER;
|
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
2009-09-04 14:28:52 +00:00
|
|
|
IHardDisk *hardDisk = NULL;
|
|
|
|
int deregister = 0;
|
|
|
|
nsresult rc;
|
2009-12-04 13:49:45 +00:00
|
|
|
int i = 0;
|
|
|
|
int j = 0;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (virUUIDParse(vol->key, uuid) < 0) {
|
|
|
|
vboxError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("Could not parse UUID from '%s'"), vol->key);
|
|
|
|
return -1;
|
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&hddIID, uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = data->vboxObj->vtbl->GetHardDisk(data->vboxObj, hddIID.value, &hardDisk);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = data->vboxObj->vtbl->FindMedium(data->vboxObj, hddIID.value,
|
|
|
|
DeviceType_HardDisk, &hardDisk);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
PRUint32 hddstate;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (hddstate != MediaState_Inaccessible) {
|
|
|
|
PRUint32 machineIdsSize = 0;
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray machineIds = VBOX_ARRAY_INITIALIZER;
|
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
vboxArrayGet(&machineIds, hardDisk, hardDisk->vtbl->imedium.GetMachineIds);
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
vboxArrayGet(&machineIds, hardDisk, hardDisk->vtbl->GetMachineIds);
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
#if VBOX_API_VERSION == 2002 && defined WIN32
|
|
|
|
/* VirtualBox 2.2 on Windows represents IIDs as GUIDs and the
|
|
|
|
* machineIds array contains direct instances of the GUID struct
|
|
|
|
* instead of pointers to the actual struct instances. But there
|
|
|
|
* is no 128bit width simple item type for a SafeArray to fit a
|
|
|
|
* GUID in. The largest simple type it 64bit width and VirtualBox
|
|
|
|
* uses two of this 64bit items to represents one GUID. Therefore,
|
|
|
|
* we devide the size of the SafeArray by two, to compensate for
|
|
|
|
* this workaround in VirtualBox */
|
|
|
|
machineIds.count /= 2;
|
|
|
|
#endif /* VBOX_API_VERSION >= 2002 */
|
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
machineIdsSize = machineIds.count;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
for (i = 0; i < machineIds.count; i++) {
|
2009-12-04 13:49:45 +00:00
|
|
|
IMachine *machine = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID machineId = VBOX_IID_INITIALIZER;
|
|
|
|
|
|
|
|
vboxIIDFromArrayItem(&machineId, &machineIds, i);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION >= 4000
|
|
|
|
rc = VBOX_OBJECT_GET_MACHINE(machineId.value, &machine);
|
|
|
|
if (NS_FAILED(rc)) {
|
|
|
|
vboxError(VIR_ERR_NO_DOMAIN, "%s",
|
|
|
|
_("no domain with matching uuid"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
rc = VBOX_SESSION_OPEN(machineId.value, machine);
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
rc = data->vboxSession->vtbl->GetMachine(data->vboxSession, &machine);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArray hddAttachments = VBOX_ARRAY_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&hddAttachments, machine,
|
|
|
|
machine->vtbl->GetHardDiskAttachments);
|
2009-12-04 13:49:45 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayGet(&hddAttachments, machine,
|
|
|
|
machine->vtbl->GetMediumAttachments);
|
2009-12-04 13:49:45 +00:00
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
2010-12-16 22:05:48 +00:00
|
|
|
for (j = 0; j < hddAttachments.count; j++) {
|
|
|
|
IHardDiskAttachment *hddAttachment = hddAttachments.items[j];
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (hddAttachment) {
|
|
|
|
IHardDisk *hdd = NULL;
|
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
rc = hddAttachment->vtbl->GetHardDisk(hddAttachment, &hdd);
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
rc = hddAttachment->vtbl->GetMedium(hddAttachment, &hdd);
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
|
|
|
if (NS_SUCCEEDED(rc) && hdd) {
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIID iid = VBOX_IID_INITIALIZER;
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = VBOX_MEDIUM_FUNC_ARG1(hdd, GetId, &iid.value);
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("HardDisk (to delete) UUID", hddIID.value);
|
|
|
|
DEBUGIID("HardDisk (currently processing) UUID", iid.value);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (vboxIIDIsEqual(&hddIID, &iid)) {
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUnichar *controller = NULL;
|
|
|
|
PRInt32 port = 0;
|
|
|
|
PRInt32 device = 0;
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("Found HardDisk to delete, UUID", hddIID.value);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
hddAttachment->vtbl->GetController(hddAttachment, &controller);
|
|
|
|
hddAttachment->vtbl->GetPort(hddAttachment, &port);
|
|
|
|
hddAttachment->vtbl->GetDevice(hddAttachment, &device);
|
|
|
|
|
|
|
|
#if VBOX_API_VERSION < 3001
|
|
|
|
rc = machine->vtbl->DetachHardDisk(machine, controller, port, device);
|
|
|
|
#else /* VBOX_API_VERSION >= 3001 */
|
|
|
|
rc = machine->vtbl->DetachDevice(machine, controller, port, device);
|
|
|
|
#endif /* VBOX_API_VERSION >= 3001 */
|
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
rc = machine->vtbl->SaveSettings(machine);
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG0("saving machine settings");
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
deregister++;
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("deregistering hdd:%d", deregister);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (controller)
|
|
|
|
VBOX_UTF16_FREE(controller);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&iid);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_RELEASE(hdd);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayRelease(&hddAttachments);
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_RELEASE(machine);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2010-12-27 22:35:30 +00:00
|
|
|
VBOX_SESSION_CLOSE();
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2010-12-23 16:25:56 +00:00
|
|
|
|
|
|
|
vboxIIDUnalloc(&machineId);
|
2009-12-04 13:49:45 +00:00
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-16 22:05:48 +00:00
|
|
|
vboxArrayUnalloc(&machineIds);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (machineIdsSize == 0 || machineIdsSize == deregister) {
|
|
|
|
IProgress *progress = NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
rc = hardDisk->vtbl->DeleteStorage(hardDisk, &progress);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc) && progress) {
|
|
|
|
progress->vtbl->WaitForCompletion(progress, -1);
|
|
|
|
VBOX_RELEASE(progress);
|
2010-12-23 16:25:56 +00:00
|
|
|
DEBUGIID("HardDisk deleted, UUID", hddIID.value);
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = 0;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDisk);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&hddIID);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2009-09-04 14:28:52 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int vboxStorageVolGetInfo(virStorageVolPtr vol, virStorageVolInfoPtr info) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(vol->conn, int, -1);
|
2009-09-04 14:28:52 +00:00
|
|
|
IHardDisk *hardDisk = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
vboxIID hddIID = VBOX_IID_INITIALIZER;
|
2009-09-04 14:28:52 +00:00
|
|
|
nsresult rc;
|
|
|
|
|
2010-01-18 08:58:57 +00:00
|
|
|
if (!info)
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (virUUIDParse(vol->key, uuid) < 0) {
|
|
|
|
vboxError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("Could not parse UUID from '%s'"), vol->key);
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2010-12-23 16:25:56 +00:00
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&hddIID, uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = data->vboxObj->vtbl->GetHardDisk(data->vboxObj, hddIID.value, &hardDisk);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = data->vboxObj->vtbl->FindMedium(data->vboxObj, hddIID.value,
|
|
|
|
DeviceType_HardDisk, &hardDisk);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
PRUint32 hddstate;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (hddstate != MediaState_Inaccessible) {
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint64 hddLogicalSize;
|
|
|
|
PRUint64 hddActualSize;
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
PRInt64 hddLogicalSize;
|
|
|
|
PRInt64 hddActualSize;
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
info->type = VIR_STORAGE_VOL_FILE;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
hardDisk->vtbl->GetLogicalSize(hardDisk, &hddLogicalSize);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2009-12-04 13:49:45 +00:00
|
|
|
info->capacity = hddLogicalSize * 1024 * 1024; /* MB => Bytes */
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
info->capacity = hddLogicalSize;
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetSize, &hddActualSize);
|
|
|
|
info->allocation = hddActualSize;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = 0;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Storage Volume Name: %s", vol->name);
|
|
|
|
VIR_DEBUG("Storage Volume Type: %s", info->type == VIR_STORAGE_VOL_BLOCK ? "Block" : "File");
|
|
|
|
VIR_DEBUG("Storage Volume Capacity: %llu", info->capacity);
|
|
|
|
VIR_DEBUG("Storage Volume Allocation: %llu", info->allocation);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDisk);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&hddIID);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2009-09-04 14:28:52 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *vboxStorageVolGetXMLDesc(virStorageVolPtr vol, unsigned int flags ATTRIBUTE_UNUSED) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(vol->conn, char *, NULL);
|
2009-09-04 14:28:52 +00:00
|
|
|
IHardDisk *hardDisk = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
vboxIID hddIID = VBOX_IID_INITIALIZER;
|
2009-09-04 14:28:52 +00:00
|
|
|
virStoragePoolDef pool;
|
|
|
|
virStorageVolDef def;
|
|
|
|
int defOk = 0;
|
|
|
|
nsresult rc;
|
|
|
|
|
|
|
|
memset(&pool, 0, sizeof(pool));
|
|
|
|
memset(&def, 0, sizeof(def));
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (virUUIDParse(vol->key, uuid) < 0) {
|
|
|
|
vboxError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("Could not parse UUID from '%s'"), vol->key);
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2010-12-23 16:25:56 +00:00
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&hddIID, uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = data->vboxObj->vtbl->GetHardDisk(data->vboxObj, hddIID.value, &hardDisk);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = data->vboxObj->vtbl->FindMedium(data->vboxObj, hddIID.value,
|
|
|
|
DeviceType_HardDisk, &hardDisk);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
PRUint32 hddstate;
|
|
|
|
|
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (NS_SUCCEEDED(rc) && hddstate != MediaState_Inaccessible) {
|
|
|
|
PRUnichar *hddFormatUtf16 = NULL;
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2009-12-04 13:49:45 +00:00
|
|
|
PRUint64 hddLogicalSize;
|
|
|
|
PRUint64 hddActualSize;
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
PRInt64 hddLogicalSize;
|
|
|
|
PRInt64 hddActualSize;
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
/* since there is currently one default pool now
|
|
|
|
* and virStorageVolDefFormat() just checks it type
|
|
|
|
* so just assign it for now, change the behaviour
|
|
|
|
* when vbox supports pools.
|
|
|
|
*/
|
|
|
|
pool.type = VIR_STORAGE_POOL_DIR;
|
|
|
|
def.type = VIR_STORAGE_VOL_FILE;
|
|
|
|
defOk = 1;
|
|
|
|
|
|
|
|
rc = hardDisk->vtbl->GetLogicalSize(hardDisk, &hddLogicalSize);
|
2010-12-27 22:35:30 +00:00
|
|
|
if (NS_SUCCEEDED(rc) && defOk) {
|
|
|
|
#if VBOX_API_VERSION < 4000
|
2009-12-04 13:49:45 +00:00
|
|
|
def.capacity = hddLogicalSize * 1024 * 1024; /* MB => Bytes */
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
def.capacity = hddLogicalSize;
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
|
|
|
} else
|
2009-12-04 13:49:45 +00:00
|
|
|
defOk = 0;
|
|
|
|
|
|
|
|
rc = VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetSize, &hddActualSize);
|
|
|
|
if (NS_SUCCEEDED(rc) && defOk)
|
|
|
|
def.allocation = hddActualSize;
|
|
|
|
else
|
|
|
|
defOk = 0;
|
|
|
|
|
|
|
|
def.name = strdup(vol->name);
|
|
|
|
if (!(def.name && defOk))
|
|
|
|
defOk = 0;
|
|
|
|
|
|
|
|
def.key = strdup(vol->key);
|
|
|
|
if (!(def.key && defOk))
|
|
|
|
defOk = 0;
|
|
|
|
|
|
|
|
rc = hardDisk->vtbl->GetFormat(hardDisk, &hddFormatUtf16);
|
|
|
|
if (NS_SUCCEEDED(rc) && defOk) {
|
|
|
|
char *hddFormatUtf8 = NULL;
|
|
|
|
|
|
|
|
VBOX_UTF16_TO_UTF8(hddFormatUtf16, &hddFormatUtf8);
|
|
|
|
if (hddFormatUtf8) {
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Storage Volume Format: %s", hddFormatUtf8);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
if (STRCASEEQ("vmdk", hddFormatUtf8))
|
|
|
|
def.target.format = VIR_STORAGE_FILE_VMDK;
|
|
|
|
else if (STRCASEEQ("vhd", hddFormatUtf8))
|
|
|
|
def.target.format = VIR_STORAGE_FILE_VPC;
|
2009-09-04 14:28:52 +00:00
|
|
|
else
|
2009-12-04 13:49:45 +00:00
|
|
|
def.target.format = VIR_STORAGE_FILE_RAW;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
/* TODO: need to add vdi to enum virStorageFileFormat {}
|
|
|
|
* and then add it here
|
|
|
|
*/
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(hddFormatUtf8);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(hddFormatUtf16);
|
|
|
|
} else {
|
|
|
|
defOk = 0;
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDisk);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&hddIID);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2009-09-04 14:28:52 +00:00
|
|
|
if (defOk)
|
2010-02-10 11:42:56 +00:00
|
|
|
ret = virStorageVolDefFormat(&pool, &def);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *vboxStorageVolGetPath(virStorageVolPtr vol) {
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_OBJECT_CHECK(vol->conn, char *, NULL);
|
2009-09-04 14:28:52 +00:00
|
|
|
IHardDisk *hardDisk = NULL;
|
2010-12-23 16:25:56 +00:00
|
|
|
unsigned char uuid[VIR_UUID_BUFLEN];
|
|
|
|
vboxIID hddIID = VBOX_IID_INITIALIZER;
|
2009-09-04 14:28:52 +00:00
|
|
|
nsresult rc;
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
if (virUUIDParse(vol->key, uuid) < 0) {
|
|
|
|
vboxError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("Could not parse UUID from '%s'"), vol->key);
|
2009-12-04 13:49:45 +00:00
|
|
|
return ret;
|
2010-12-23 16:25:56 +00:00
|
|
|
}
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDFromUUID(&hddIID, uuid);
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION < 4000
|
2010-12-23 16:25:56 +00:00
|
|
|
rc = data->vboxObj->vtbl->GetHardDisk(data->vboxObj, hddIID.value, &hardDisk);
|
2010-12-27 22:35:30 +00:00
|
|
|
#else /* VBOX_API_VERSION >= 4000 */
|
|
|
|
rc = data->vboxObj->vtbl->FindMedium(data->vboxObj, hddIID.value,
|
|
|
|
DeviceType_HardDisk, &hardDisk);
|
|
|
|
#endif /* VBOX_API_VERSION >= 4000 */
|
2009-12-04 13:49:45 +00:00
|
|
|
if (NS_SUCCEEDED(rc)) {
|
|
|
|
PRUint32 hddstate;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetState, &hddstate);
|
|
|
|
if (hddstate != MediaState_Inaccessible) {
|
|
|
|
PRUnichar *hddLocationUtf16 = NULL;
|
|
|
|
char *hddLocationUtf8 = NULL;
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_MEDIUM_FUNC_ARG1(hardDisk, GetLocation, &hddLocationUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_TO_UTF8(hddLocationUtf16, &hddLocationUtf8);
|
|
|
|
if (hddLocationUtf8) {
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
ret = strdup(hddLocationUtf8);
|
|
|
|
if (!ret)
|
2010-02-04 18:19:08 +00:00
|
|
|
virReportOOMError();
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("Storage Volume Name: %s", vol->name);
|
|
|
|
VIR_DEBUG("Storage Volume Path: %s", hddLocationUtf8);
|
|
|
|
VIR_DEBUG("Storage Volume Pool: %s", vol->pool);
|
2009-09-04 14:28:52 +00:00
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF8_FREE(hddLocationUtf8);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2009-12-04 13:49:45 +00:00
|
|
|
VBOX_UTF16_FREE(hddLocationUtf16);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
2009-12-04 13:49:45 +00:00
|
|
|
|
|
|
|
VBOX_MEDIUM_RELEASE(hardDisk);
|
2009-09-04 14:28:52 +00:00
|
|
|
}
|
|
|
|
|
2010-12-23 16:25:56 +00:00
|
|
|
vboxIIDUnalloc(&hddIID);
|
2009-12-04 13:49:45 +00:00
|
|
|
|
2009-09-04 14:28:52 +00:00
|
|
|
return ret;
|
|
|
|
}
|
2009-05-15 14:00:50 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Tables
|
|
|
|
*/
|
|
|
|
|
2009-04-17 16:09:07 +00:00
|
|
|
virDriver NAME(Driver) = {
|
|
|
|
VIR_DRV_VBOX,
|
|
|
|
"VBOX",
|
2009-06-03 13:30:19 +00:00
|
|
|
vboxOpen, /* open */
|
|
|
|
vboxClose, /* close */
|
|
|
|
NULL, /* supports_feature */
|
|
|
|
NULL, /* type */
|
|
|
|
vboxGetVersion, /* version */
|
2009-11-12 15:53:26 +00:00
|
|
|
NULL, /* libvirtVersion (impl. in libvirt.c) */
|
2009-10-23 17:01:22 +00:00
|
|
|
virGetHostname, /* getHostname */
|
2011-02-07 21:14:56 +00:00
|
|
|
NULL, /* getSysinfo */
|
2009-06-03 13:30:19 +00:00
|
|
|
vboxGetMaxVcpus, /* getMaxVcpus */
|
|
|
|
nodeGetInfo, /* nodeGetInfo */
|
|
|
|
vboxGetCapabilities, /* getCapabilities */
|
|
|
|
vboxListDomains, /* listDomains */
|
|
|
|
vboxNumOfDomains, /* numOfDomains */
|
|
|
|
vboxDomainCreateXML, /* domainCreateXML */
|
|
|
|
vboxDomainLookupByID, /* domainLookupByID */
|
|
|
|
vboxDomainLookupByUUID, /* domainLookupByUUID */
|
|
|
|
vboxDomainLookupByName, /* domainLookupByName */
|
|
|
|
vboxDomainSuspend, /* domainSuspend */
|
|
|
|
vboxDomainResume, /* domainResume */
|
|
|
|
vboxDomainShutdown, /* domainShutdown */
|
|
|
|
vboxDomainReboot, /* domainReboot */
|
|
|
|
vboxDomainDestroy, /* domainDestroy */
|
|
|
|
vboxDomainGetOSType, /* domainGetOSType */
|
|
|
|
NULL, /* domainGetMaxMemory */
|
|
|
|
NULL, /* domainSetMaxMemory */
|
|
|
|
vboxDomainSetMemory, /* domainSetMemory */
|
2011-03-02 08:07:48 +00:00
|
|
|
NULL, /* domainSetMemoryFlags */
|
2011-02-22 05:31:57 +00:00
|
|
|
NULL, /* domainSetMemoryParameters */
|
|
|
|
NULL, /* domainGetMemoryParameters */
|
|
|
|
NULL, /* domainSetBlkioParameters */
|
|
|
|
NULL, /* domainGetBlkioParameters */
|
2009-06-03 13:30:19 +00:00
|
|
|
vboxDomainGetInfo, /* domainGetInfo */
|
|
|
|
vboxDomainSave, /* domainSave */
|
|
|
|
NULL, /* domainRestore */
|
|
|
|
NULL, /* domainCoreDump */
|
2009-09-02 14:08:14 +00:00
|
|
|
vboxDomainSetVcpus, /* domainSetVcpus */
|
vcpu: make old API trivially wrap to new API
Note - this wrapping is completely mechanical; the old API will
function identically, since the new API validates that the exact
same flags are provided by the old API. On a per-driver basis,
it may make sense to have the old API pass a different set of flags,
but that should be done in the per-driver patch that implements
the full range of flag support in the new API.
* src/esx/esx_driver.c (esxDomainSetVcpus, escDomainGetMaxVpcus):
Move guts...
(esxDomainSetVcpusFlags, esxDomainGetVcpusFlags): ...to new
functions.
(esxDriver): Trivially support the new API.
* src/openvz/openvz_driver.c (openvzDomainSetVcpus)
(openvzDomainSetVcpusFlags, openvzDomainGetMaxVcpus)
(openvzDomainGetVcpusFlags, openvzDriver): Likewise.
* src/phyp/phyp_driver.c (phypDomainSetCPU)
(phypDomainSetVcpusFlags, phypGetLparCPUMAX)
(phypDomainGetVcpusFlags, phypDriver): Likewise.
* src/qemu/qemu_driver.c (qemudDomainSetVcpus)
(qemudDomainSetVcpusFlags, qemudDomainGetMaxVcpus)
(qemudDomainGetVcpusFlags, qemuDriver): Likewise.
* src/test/test_driver.c (testSetVcpus, testDomainSetVcpusFlags)
(testDomainGetMaxVcpus, testDomainGetVcpusFlags, testDriver):
Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainSetVcpus)
(vboxDomainSetVcpusFlags, virDomainGetMaxVcpus)
(virDomainGetVcpusFlags, virDriver): Likewise.
* src/xen/xen_driver.c (xenUnifiedDomainSetVcpus)
(xenUnifiedDomainSetVcpusFlags, xenUnifiedDomainGetMaxVcpus)
(xenUnifiedDomainGetVcpusFlags, xenUnifiedDriver): Likewise.
* src/xenapi/xenapi_driver.c (xenapiDomainSetVcpus)
(xenapiDomainSetVcpusFlags, xenapiDomainGetMaxVcpus)
(xenapiDomainGetVcpusFlags, xenapiDriver): Likewise.
(xenapiError): New helper macro.
2010-09-27 22:37:53 +00:00
|
|
|
vboxDomainSetVcpusFlags, /* domainSetVcpusFlags */
|
|
|
|
vboxDomainGetVcpusFlags, /* domainGetVcpusFlags */
|
2009-06-03 13:30:19 +00:00
|
|
|
NULL, /* domainPinVcpu */
|
|
|
|
NULL, /* domainGetVcpus */
|
2009-09-02 14:08:14 +00:00
|
|
|
vboxDomainGetMaxVcpus, /* domainGetMaxVcpus */
|
2009-06-03 13:30:19 +00:00
|
|
|
NULL, /* domainGetSecurityLabel */
|
|
|
|
NULL, /* nodeGetSecurityModel */
|
|
|
|
vboxDomainDumpXML, /* domainDumpXML */
|
2009-09-06 14:03:06 +00:00
|
|
|
NULL, /* domainXMLFromNative */
|
|
|
|
NULL, /* domainXMLToNative */
|
2009-06-03 13:30:19 +00:00
|
|
|
vboxListDefinedDomains, /* listDefinedDomains */
|
|
|
|
vboxNumOfDefinedDomains, /* numOfDefinedDomains */
|
|
|
|
vboxDomainCreate, /* domainCreate */
|
2010-06-10 15:55:36 +00:00
|
|
|
vboxDomainCreateWithFlags, /* domainCreateWithFlags */
|
2009-06-03 13:30:19 +00:00
|
|
|
vboxDomainDefineXML, /* domainDefineXML */
|
|
|
|
vboxDomainUndefine, /* domainUndefine */
|
|
|
|
vboxDomainAttachDevice, /* domainAttachDevice */
|
2010-01-14 01:44:26 +00:00
|
|
|
vboxDomainAttachDeviceFlags, /* domainAttachDeviceFlags */
|
2009-06-03 13:30:19 +00:00
|
|
|
vboxDomainDetachDevice, /* domainDetachDevice */
|
2010-01-14 01:44:26 +00:00
|
|
|
vboxDomainDetachDeviceFlags, /* domainDetachDeviceFlags */
|
2010-03-22 13:13:53 +00:00
|
|
|
vboxDomainUpdateDeviceFlags, /* domainUpdateDeviceFlags */
|
2009-06-03 13:30:19 +00:00
|
|
|
NULL, /* domainGetAutostart */
|
|
|
|
NULL, /* domainSetAutostart */
|
|
|
|
NULL, /* domainGetSchedulerType */
|
|
|
|
NULL, /* domainGetSchedulerParameters */
|
|
|
|
NULL, /* domainSetSchedulerParameters */
|
|
|
|
NULL, /* domainMigratePrepare */
|
|
|
|
NULL, /* domainMigratePerform */
|
|
|
|
NULL, /* domainMigrateFinish */
|
|
|
|
NULL, /* domainBlockStats */
|
|
|
|
NULL, /* domainInterfaceStats */
|
2009-12-20 12:28:42 +00:00
|
|
|
NULL, /* domainMemoryStats */
|
2009-06-03 13:30:19 +00:00
|
|
|
NULL, /* domainBlockPeek */
|
|
|
|
NULL, /* domainMemoryPeek */
|
Internal driver API infrastructure for virDomainGetBlockInfo
This defines the internal driver API and stubs out each driver
* src/driver.h: Define virDrvDomainGetBlockInfo signature
* src/libvirt.c, src/libvirt_public.syms: Glue public API to drivers
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/opennebula/one_driver.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xenapi/xenapi_driver.c: Stub out driver
2010-04-27 19:27:34 +00:00
|
|
|
NULL, /* domainGetBlockInfo */
|
2009-06-03 14:32:18 +00:00
|
|
|
nodeGetCellsFreeMemory, /* nodeGetCellsFreeMemory */
|
|
|
|
nodeGetFreeMemory, /* getFreeMemory */
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION == 2002 || VBOX_API_VERSION == 4000
|
2009-06-03 13:30:19 +00:00
|
|
|
NULL, /* domainEventRegister */
|
|
|
|
NULL, /* domainEventDeregister */
|
2009-07-24 16:12:16 +00:00
|
|
|
#else
|
|
|
|
vboxDomainEventRegister, /* domainEventRegister */
|
|
|
|
vboxDomainEventDeregister, /* domainEventDeregister */
|
|
|
|
#endif
|
2009-06-03 13:30:19 +00:00
|
|
|
NULL, /* domainMigratePrepare2 */
|
|
|
|
NULL, /* domainMigrateFinish2 */
|
|
|
|
NULL, /* nodeDeviceDettach */
|
|
|
|
NULL, /* nodeDeviceReAttach */
|
|
|
|
NULL, /* nodeDeviceReset */
|
2009-09-30 10:51:54 +00:00
|
|
|
NULL, /* domainMigratePrepareTunnel */
|
2010-03-23 18:00:38 +00:00
|
|
|
vboxIsEncrypted, /* isEncrypted */
|
|
|
|
vboxIsSecure, /* isSecure */
|
|
|
|
vboxDomainIsActive, /* domainIsActive */
|
|
|
|
vboxDomainIsPersistent, /* domainIsPersistent */
|
2010-11-24 07:43:15 +00:00
|
|
|
vboxDomainIsUpdated, /* domainIsUpdated */
|
2009-12-18 13:59:39 +00:00
|
|
|
NULL, /* cpuCompare */
|
2010-02-02 11:34:01 +00:00
|
|
|
NULL, /* cpuBaseline */
|
Stub out internal driver entry points for job processing
The internal glue layer for the new pubic API
* src/driver.h: Define internal driver API contract
* src/libvirt.c, src/libvirt_public.syms: Wire up public
API to internal driver API
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/opennebula/one_driver.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/qemu/qemu_driver.c, src/remote/remote_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c: Stub new entry point
2010-02-03 11:32:24 +00:00
|
|
|
NULL, /* domainGetJobInfo */
|
Wire up internal entry points for virDomainAbortJob API
This provides the internal glue for the driver API
* src/driver.h: Internal API contract
* src/libvirt.c, src/libvirt_public.syms: Connect public API
to driver API
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/opennebula/one_driver.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/qemu/qemu_driver.c, src/remote/remote_driver.c,
src/test/test_driver.c src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c: Stub out entry points
2010-02-04 16:16:35 +00:00
|
|
|
NULL, /* domainAbortJob */
|
2010-03-12 13:55:08 +00:00
|
|
|
NULL, /* domainMigrateSetMaxDowntime */
|
2010-12-27 22:35:30 +00:00
|
|
|
#if VBOX_API_VERSION == 2002 || VBOX_API_VERSION == 4000
|
Introduce a new public API for domain events
The current API for domain events has a number of problems
- Only allows for domain lifecycle change events
- Does not allow the same callback to be registered multiple times
- Does not allow filtering of events to a specific domain
This introduces a new more general purpose domain events API
typedef enum {
VIR_DOMAIN_EVENT_ID_LIFECYCLE = 0, /* virConnectDomainEventCallback */
...more events later..
}
int virConnectDomainEventRegisterAny(virConnectPtr conn,
virDomainPtr dom, /* Optional, to filter */
int eventID,
virConnectDomainEventGenericCallback cb,
void *opaque,
virFreeCallback freecb);
int virConnectDomainEventDeregisterAny(virConnectPtr conn,
int callbackID);
Since different event types can received different data in the callback,
the API is defined with a generic callback. Specific events will each
have a custom signature for their callback. Thus when registering an
event it is neccessary to cast the callback to the generic signature
eg
int myDomainEventCallback(virConnectPtr conn,
virDomainPtr dom,
int event,
int detail,
void *opaque)
{
...
}
virConnectDomainEventRegisterAny(conn, NULL,
VIR_DOMAIN_EVENT_ID_LIFECYCLE,
VIR_DOMAIN_EVENT_CALLBACK(myDomainEventCallback)
NULL, NULL);
The VIR_DOMAIN_EVENT_CALLBACK() macro simply does a "bad" cast
to the generic signature
* include/libvirt/libvirt.h.in: Define new APIs for registering
domain events
* src/driver.h: Internal driver entry points for new events APIs
* src/libvirt.c: Wire up public API to driver API for events APIs
* src/libvirt_public.syms: Export new APIs
* src/esx/esx_driver.c, src/lxc/lxc_driver.c, src/opennebula/one_driver.c,
src/openvz/openvz_driver.c, src/phyp/phyp_driver.c,
src/qemu/qemu_driver.c, src/remote/remote_driver.c,
src/test/test_driver.c, src/uml/uml_driver.c,
src/vbox/vbox_tmpl.c, src/xen/xen_driver.c,
src/xenapi/xenapi_driver.c: Stub out new API entries
2010-03-18 13:01:48 +00:00
|
|
|
NULL, /* domainEventRegisterAny */
|
|
|
|
NULL, /* domainEventDeregisterAny */
|
2010-03-18 14:47:07 +00:00
|
|
|
#else
|
|
|
|
vboxDomainEventRegisterAny, /* domainEventRegisterAny */
|
|
|
|
vboxDomainEventDeregisterAny, /* domainEventDeregisterAny */
|
|
|
|
#endif
|
2010-04-01 08:46:28 +00:00
|
|
|
NULL, /* domainManagedSave */
|
|
|
|
NULL, /* domainHasManagedSaveImage */
|
|
|
|
NULL, /* domainManagedSaveRemove */
|
2010-03-26 12:40:34 +00:00
|
|
|
vboxDomainSnapshotCreateXML, /* domainSnapshotCreateXML */
|
|
|
|
vboxDomainSnapshotDumpXML, /* domainSnapshotDumpXML */
|
|
|
|
vboxDomainSnapshotNum, /* domainSnapshotNum */
|
|
|
|
vboxDomainSnapshotListNames, /* domainSnapshotListNames */
|
|
|
|
vboxDomainSnapshotLookupByName, /* domainSnapshotLookupByName */
|
|
|
|
vboxDomainHasCurrentSnapshot, /* domainHasCurrentSnapshot */
|
|
|
|
vboxDomainSnapshotCurrent, /* domainSnapshotCurrent */
|
|
|
|
vboxDomainRevertToSnapshot, /* domainRevertToSnapshot */
|
|
|
|
vboxDomainSnapshotDelete, /* domainSnapshotDelete */
|
2010-04-13 18:02:46 +00:00
|
|
|
NULL, /* qemuDomainMonitorCommand */
|
Introduce a virDomainOpenConsole API
To enable virsh console (or equivalent) to be used remotely
it is necessary to provide remote access to the /dev/pts/XXX
pseudo-TTY associated with the console/serial/parallel device
in the guest. The virStream API provide a bi-directional I/O
stream capability that can be used for this purpose. This
patch thus introduces a virDomainOpenConsole API that uses
the stream APIs.
* src/libvirt.c, src/libvirt_public.syms,
include/libvirt/libvirt.h.in, src/driver.h: Define the
new virDomainOpenConsole API
* src/esx/esx_driver.c, src/lxc/lxc_driver.c,
src/opennebula/one_driver.c, src/openvz/openvz_driver.c,
src/phyp/phyp_driver.c, src/qemu/qemu_driver.c,
src/remote/remote_driver.c, src/test/test_driver.c,
src/uml/uml_driver.c, src/vbox/vbox_tmpl.c,
src/xen/xen_driver.c, src/xenapi/xenapi_driver.c: Stub
API entry point
2010-07-23 12:34:31 +00:00
|
|
|
NULL, /* domainOpenConsole */
|
2009-04-17 16:09:07 +00:00
|
|
|
};
|
2009-05-15 14:00:50 +00:00
|
|
|
|
|
|
|
virNetworkDriver NAME(NetworkDriver) = {
|
|
|
|
"VBOX",
|
|
|
|
.open = vboxNetworkOpen,
|
|
|
|
.close = vboxNetworkClose,
|
|
|
|
.numOfNetworks = vboxNumOfNetworks,
|
|
|
|
.listNetworks = vboxListNetworks,
|
|
|
|
.numOfDefinedNetworks = vboxNumOfDefinedNetworks,
|
|
|
|
.listDefinedNetworks = vboxListDefinedNetworks,
|
|
|
|
.networkLookupByUUID = vboxNetworkLookupByUUID,
|
|
|
|
.networkLookupByName = vboxNetworkLookupByName,
|
|
|
|
.networkCreateXML = vboxNetworkCreateXML,
|
|
|
|
.networkDefineXML = vboxNetworkDefineXML,
|
|
|
|
.networkUndefine = vboxNetworkUndefine,
|
|
|
|
.networkCreate = vboxNetworkCreate,
|
|
|
|
.networkDestroy = vboxNetworkDestroy,
|
|
|
|
.networkDumpXML = vboxNetworkDumpXML,
|
|
|
|
.networkGetBridgeName = NULL,
|
|
|
|
.networkGetAutostart = NULL,
|
|
|
|
.networkSetAutostart = NULL
|
|
|
|
};
|
2009-09-04 14:28:52 +00:00
|
|
|
|
|
|
|
virStorageDriver NAME(StorageDriver) = {
|
|
|
|
.name = "VBOX",
|
|
|
|
.open = vboxStorageOpen,
|
|
|
|
.close = vboxStorageClose,
|
|
|
|
.numOfPools = vboxStorageNumOfPools,
|
|
|
|
.listPools = vboxStorageListPools,
|
|
|
|
.numOfDefinedPools = NULL,
|
|
|
|
.listDefinedPools = NULL,
|
|
|
|
.findPoolSources = NULL,
|
|
|
|
.poolLookupByName = vboxStoragePoolLookupByName,
|
|
|
|
.poolLookupByUUID = NULL,
|
|
|
|
.poolLookupByVolume = NULL,
|
|
|
|
.poolCreateXML = NULL,
|
|
|
|
.poolDefineXML = NULL,
|
|
|
|
.poolBuild = NULL,
|
|
|
|
.poolUndefine = NULL,
|
|
|
|
.poolCreate = NULL,
|
|
|
|
.poolDestroy = NULL,
|
|
|
|
.poolDelete = NULL,
|
|
|
|
.poolRefresh = NULL,
|
|
|
|
.poolGetInfo = NULL,
|
|
|
|
.poolGetXMLDesc = NULL,
|
|
|
|
.poolGetAutostart = NULL,
|
|
|
|
.poolSetAutostart = NULL,
|
|
|
|
.poolNumOfVolumes = vboxStoragePoolNumOfVolumes,
|
|
|
|
.poolListVolumes = vboxStoragePoolListVolumes,
|
|
|
|
|
|
|
|
.volLookupByName = vboxStorageVolLookupByName,
|
|
|
|
.volLookupByKey = vboxStorageVolLookupByKey,
|
|
|
|
.volLookupByPath = vboxStorageVolLookupByPath,
|
|
|
|
.volCreateXML = vboxStorageVolCreateXML,
|
|
|
|
.volCreateXMLFrom = NULL,
|
|
|
|
.volDelete = vboxStorageVolDelete,
|
|
|
|
.volGetInfo = vboxStorageVolGetInfo,
|
|
|
|
.volGetXMLDesc = vboxStorageVolGetXMLDesc,
|
2009-12-04 13:49:45 +00:00
|
|
|
.volGetPath = vboxStorageVolGetPath
|
2009-09-04 14:28:52 +00:00
|
|
|
};
|