/** @file * * VBox frontends: VBoxManage (command-line interface) * * VBoxManage is VirtualBox's command-line interface. This is its rather * long source. */ /* * Copyright (C) 2006 InnoTek Systemberatung GmbH * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License as published by the Free Software Foundation, * in version 2 as it comes in the "COPYING" file of the VirtualBox OSE * distribution. VirtualBox OSE is distributed in the hope that it will * be useful, but WITHOUT ANY WARRANTY of any kind. * * If you received this file as part of a commercial VirtualBox * distribution, then only the terms of your commercial VirtualBox * license agreement apply instead of the previous paragraph. */ /******************************************************************************* * Header Files * *******************************************************************************/ #include #include #include #include #include #include #define CFGLDR_HAVE_COM #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "VBoxManage.h" using namespace com; /* missing XPCOM <-> COM wrappers */ #ifndef STDMETHOD_ # define STDMETHOD_(ret, meth) NS_IMETHOD_(ret) meth #endif #ifndef NS_GET_IID # define NS_GET_IID(I) IID_##I #endif #ifndef __WIN__ #define IUnknown nsISupports #endif /** command handler type */ typedef DECLCALLBACK(int) FNHANDLER(int argc, char *argv[], ComPtr aVirtualBox, ComPtr aSession); typedef FNHANDLER *PFNHANDLER; /** * Quick IUSBDevice implementation for detaching / attaching * devices to the USB Controller. */ class MyUSBDevice : public IUSBDevice { public: // public initializer/uninitializer for internal purposes only MyUSBDevice(uint16_t a_u16VendorId, uint16_t a_u16ProductId, uint16_t a_bcdRevision, uint64_t a_u64SerialHash, const char *a_pszComment) : m_usVendorId(a_u16VendorId), m_usProductId(a_u16ProductId), m_bcdRevision(a_bcdRevision), m_u64SerialHash(a_u64SerialHash), m_bstrComment(a_pszComment), m_cRefs(0) { } STDMETHOD_(ULONG, AddRef)(void) { return ASMAtomicIncU32(&m_cRefs); } STDMETHOD_(ULONG, Release)(void) { ULONG cRefs = ASMAtomicDecU32(&m_cRefs); if (!cRefs) delete this; return cRefs; } STDMETHOD(QueryInterface)(const IID &iid, void **ppvObject) { Guid guid(iid); if (guid == Guid(NS_GET_IID(IUnknown))) *ppvObject = (IUnknown *)this; else if (guid == Guid(NS_GET_IID(IUSBDevice))) *ppvObject = (IUSBDevice *)this; else return E_NOINTERFACE; AddRef(); return S_OK; } STDMETHOD(COMGETTER(Id))(GUIDPARAMOUT a_pId) { return E_NOTIMPL; } STDMETHOD(COMGETTER(VendorId))(USHORT *a_pusVendorId) { *a_pusVendorId = m_usVendorId; return S_OK; } STDMETHOD(COMGETTER(ProductId))(USHORT *a_pusProductId) { *a_pusProductId = m_usProductId; return S_OK; } STDMETHOD(COMGETTER(Revision))(USHORT *a_pusRevision) { *a_pusRevision = m_bcdRevision; return S_OK; } STDMETHOD(COMGETTER(SerialHash))(ULONG64 *a_pullSerialHash) { *a_pullSerialHash = m_u64SerialHash; return S_OK; } STDMETHOD(COMGETTER(Manufacturer))(BSTR *a_pManufacturer) { return E_NOTIMPL; } STDMETHOD(COMGETTER(Product))(BSTR *a_pProduct) { return E_NOTIMPL; } STDMETHOD(COMGETTER(SerialNumber))(BSTR *a_pSerialNumber) { return E_NOTIMPL; } STDMETHOD(COMGETTER(Address))(BSTR *a_pAddress) { return E_NOTIMPL; } private: /** The vendor id of this USB device. */ USHORT m_usVendorId; /** The product id of this USB device. */ USHORT m_usProductId; /** The product revision number of this USB device. * (high byte = integer; low byte = decimal) */ USHORT m_bcdRevision; /** The USB serial hash of the device. */ uint64_t m_u64SerialHash; /** The user comment string. */ Bstr m_bstrComment; /** Reference counter. */ uint32_t volatile m_cRefs; }; // types /////////////////////////////////////////////////////////////////////////////// template class Nullable { public: Nullable() : mIsNull (true) {} Nullable (const T &aValue, bool aIsNull = false) : mIsNull (aIsNull), mValue (aValue) {} bool isNull() const { return mIsNull; }; void setNull (bool aIsNull = true) { mIsNull = aIsNull; } operator const T&() const { return mValue; } Nullable &operator= (const T &aValue) { mValue = aValue; mIsNull = false; return *this; } private: bool mIsNull; T mValue; }; /** helper structure to encapsulate USB filter manipulation commands */ struct USBFilterCmd { struct USBFilter { USBFilter () : mAction (USBDeviceFilterAction_InvalidUSBDeviceFilterAction) {} Bstr mName; Nullable mActive; Bstr mVendorId; Bstr mProductId; Bstr mRevision; Bstr mManufacturer; Bstr mProduct; Bstr mRemote; Bstr mSerialNumber; USBDeviceFilterAction_T mAction; }; enum Action { Invalid, Add, Modify, Remove }; USBFilterCmd() : mAction (Invalid), mIndex (0), mGlobal (false) {} Action mAction; ULONG mIndex; /** flag whether the command target is a global filter */ bool mGlobal; /** machine this command is targeted at (null for global filters) */ ComPtr mMachine; USBFilter mFilter; }; // funcs /////////////////////////////////////////////////////////////////////////////// static void printUsage(USAGECATEGORY u64Cmd) { #ifdef __LINUX__ bool fLinux = true; #else bool fLinux = false; #endif #ifdef __WIN__ bool fWin = true; #else bool fWin = false; #endif #ifdef __DARWIN__ bool fDarwin = true; #else bool fDarwin = false; #endif #ifdef VBOX_VRDP bool fVRDP = true; #else bool fVRDP = false; #endif if (u64Cmd == USAGE_DUMPOPTS) { fLinux = true; fWin = true; fVRDP = true; u64Cmd = USAGE_ALL; } RTPrintf("Usage:\n" "\n"); if (u64Cmd & USAGE_LIST) { RTPrintf("VBoxManage list vms|ostypes|hostdvds|hostfloppies|"); if (fWin) RTPrintf( "hostifs|"); RTPrintf( "\n" " hdds|dvds|floppies|usbhost|usbfilters|\n" " systemproperties\n" "\n"); } if (u64Cmd & USAGE_SHOWVMINFO) { RTPrintf("VBoxManage showvminfo |\n" " [-details]\n" "\n"); } if (u64Cmd & USAGE_REGISTERVM) { RTPrintf("VBoxManage registervm \n" "\n"); } if (u64Cmd & USAGE_UNREGISTERVM) { RTPrintf("VBoxManage unregistervm |\n" " [-delete]\n" "\n"); } if (u64Cmd & USAGE_CREATEVM) { RTPrintf("VBoxManage createvm -name \n" " [-register]\n" " [-basefolder | -settingsfile ]\n" " \n" "\n"); } if (u64Cmd & USAGE_MODIFYVM) { RTPrintf("VBoxManage modifyvm \n" " [-name ]\n" " [-ostype ]\n" " [-memory ]\n" " [-vram ]\n" " [-acpi on|off]\n" " [-ioapic on|off]\n" " [-hwvirtex on|off|default]\n" " [-bioslogofadein on|off]\n" " [-bioslogofadeout on|off]\n" " [-bioslogodisplaytime ]\n" " [-bioslogoimagepath ]\n" " [-biosbootmenu ]\n" " [-boot<1-4> none|floppy|dvd|disk|net>]\n" " [-hd none||]\n" " [-dvd none|||host:]\n" " [-dvdpassthrough on|off]\n" " [-floppy disabled|empty||\n" " |host:]\n" " [-nic<1-N> none|null|nat|hostif|intnet]\n" " [-nictype<1-N> Am79C970A|Am79C973]\n" " [-cableconnected<1-N> on|off]\n" " [-nictrace<1-N> on|off]\n" " [-nictracefile<1-N> ]\n" " [-hostifdev<1-N> none|]\n" " [-intnet<1-N> network]\n" " [-macaddress<1-N> auto|\n"); if (fLinux) { RTPrintf(" [-tapsetup<1-N> none|]\n" " [-tapterminate<1-N> none|]\n"); } RTPrintf(" [-audio none|null"); if (fWin) { RTPrintf( "|winmm|dsound"); } if (fLinux) { RTPrintf( "|oss" #ifdef VBOX_WITH_ALSA "|alsa" #endif ); } if (fDarwin) { RTPrintf( "|coreaudio"); } RTPrintf( "]\n"); RTPrintf(" [-clipboard disabled|hosttoguest|guesttohost|\n" " bidirectional]\n"); if (fVRDP) { RTPrintf(" [-vrdp on|off]\n" " [-vrdpport default|]\n" " [-vrdpaddress ]\n" " [-vrdpauthtype null|external|guest]\n"); } RTPrintf(" [-usb on|off]\n" " [-snapshotfolder default|]\n" "\n"); } if (u64Cmd & USAGE_STARTVM) { RTPrintf("VBoxManage startvm |\n" " [-type gui|vrdp]\n" "\n"); } if (u64Cmd & USAGE_CONTROLVM) { RTPrintf("VBoxManage controlvm |\n" " pause|resume|reset|poweroff|savestate|\n" " acpipowerbutton |\n" " setlinkstate<1-4> on|off |\n" " usbattach |
|\n" " usbdetach |
|\n" " dvdattach none|||host: |\n" " floppyattach none|||host: |\n" " setvideomodehint |\n" " setcredentials \n" " [-allowlocallogon ]\n" "\n"); } if (u64Cmd & USAGE_DISCARDSTATE) { RTPrintf("VBoxManage discardstate |\n" "\n"); } if (u64Cmd & USAGE_SNAPSHOT) { RTPrintf("VBoxManage snapshot |\n" " take [-desc ] |\n" " discard | |\n" " discardcurrent -state|-all |\n" " edit ||-current\n" " [-newname ]\n" " [-newdesc ] |\n" " showvminfo |\n" "\n"); } if (u64Cmd & USAGE_REGISTERIMAGE) { RTPrintf("VBoxManage registerimage disk|dvd|floppy \n" " [-type normal|immutable|writethrough] (disk only)\n" "\n"); } if (u64Cmd & USAGE_UNREGISTERIMAGE) { RTPrintf("VBoxManage unregisterimage disk|dvd|floppy |\n" "\n"); } if (u64Cmd & USAGE_SHOWVDIINFO) { RTPrintf("VBoxManage showvdiinfo |\n" "\n"); } if (u64Cmd & USAGE_CREATEVDI) { RTPrintf("VBoxManage createvdi -filename \n" " -size \n" " [-static]\n" " [-comment ]\n" " [-register]\n" " [-type normal|writethrough] (default: normal)\n" "\n"); } if (u64Cmd & USAGE_MODIFYVDI) { RTPrintf("VBoxManage modifyvdi |\n" #if 0 /* doesn't currently work */ " settype normal|writethrough|immutable |\n" #endif " compact\n" "\n"); } if (u64Cmd & USAGE_CLONEVDI) { RTPrintf("VBoxManage clonevdi | \n" "\n"); } if (u64Cmd & USAGE_CONVERTDD) { RTPrintf("VBoxManage convertdd \n" "VBoxManage convertdd stdin \n" "\n"); } if (u64Cmd & USAGE_ADDISCSIDISK) { RTPrintf("VBoxManage addiscsidisk -server |\n" " -target \n" " [-port ]\n" " [-lun ]\n" " [-encodedlun ]\n" " [-username ]\n" " [-password ]\n" " [-comment ]\n" "\n"); } if (u64Cmd & USAGE_CREATEHOSTIF && fWin) { RTPrintf("VBoxManage createhostif \n" "\n"); } if (u64Cmd & USAGE_REMOVEHOSTIF && fWin) { RTPrintf("VBoxManage removehostif |\n" "\n"); } if (u64Cmd & USAGE_GETEXTRADATA) { RTPrintf("VBoxManage getextradata global||\n" " |enumerate\n" "\n"); } if (u64Cmd & USAGE_SETEXTRADATA) { RTPrintf("VBoxManage setextradata global||\n" " \n" " [] (no value deletes key)\n" "\n"); } if (u64Cmd & USAGE_SETPROPERTY) { RTPrintf("VBoxManage setproperty vdifolder default| |\n" " machinefolder default| |\n" " vrdpauthlibrary default| |\n" " hwvirtexenabled yes|no\n" "\n"); } if (u64Cmd & USAGE_USBFILTER_ADD) { RTPrintf("VBoxManage usbfilter add \n" " -target ||global\n" " -name \n" " -action ignore|hold (global filters only)\n" " [-active yes|no] (yes)\n" " [-vendorid ] (null)\n" " [-productid ] (null)\n" " [-revision ] (null)\n" " [-manufacturer ] (null)\n" " [-product ] (null)\n" " [-remote yes|no] (null, VM filters only)\n" " [-serialnumber ] (null)\n" "\n"); } if (u64Cmd & USAGE_USBFILTER_MODIFY) { RTPrintf("VBoxManage usbfilter modify \n" " -target ||global\n" " [-name ]\n" " [-action ignore|hold] (global filters only)\n" " [-active yes|no]\n" " [-vendorid |\"\"]\n" " [-productid |\"\"]\n" " [-revision |\"\"]\n" " [-manufacturer |\"\"]\n" " [-product |\"\"]\n" " [-remote yes|no] (null, VM filters only)\n" " [-serialnumber |\"\"]\n" "\n"); } if (u64Cmd & USAGE_USBFILTER_REMOVE) { RTPrintf("VBoxManage usbfilter remove \n" " -target ||global\n" "\n"); } if (u64Cmd & USAGE_SHAREDFOLDER_ADD) { RTPrintf("VBoxManage sharedfolder add |\n" " -name -hostpath \n" " [-transient]\n" "\n"); } if (u64Cmd & USAGE_SHAREDFOLDER_REMOVE) { RTPrintf("VBoxManage sharedfolder remove |\n" " -name [-transient]\n" "\n"); } if (u64Cmd & USAGE_UPDATESETTINGS) { RTPrintf("VBoxManage updatesettings [|] [-apply]\n" " [-nobackup] [-skipinvalid]\n" "\n"); } } /** * Print a usage synopsis and the syntax error message. */ int errorSyntax(USAGECATEGORY u64Cmd, const char *pszFormat, ...) { va_list args; if (fInternalMode) printUsageInternal(u64Cmd); else printUsage(u64Cmd); va_start(args, pszFormat); RTPrintf("\n" "Syntax error: %N\n", pszFormat, &args); va_end(args); return 1; } /** * Print an error message without the syntax stuff. */ int errorArgument(const char *pszFormat, ...) { va_list args; va_start(args, pszFormat); RTPrintf("error: %N\n", pszFormat, &args); va_end(args); return 1; } /** * Print out progress on the console */ static void showProgress(ComPtr progress) { BOOL fCompleted; LONG currentPercent, lastPercent = 0; RTPrintf("0%%..."); RTStrmFlush(g_pStdOut); while (SUCCEEDED(progress->COMGETTER(Completed(&fCompleted)))) { progress->COMGETTER(Percent(¤tPercent)); /* did we cross a 10% mark? */ if (((currentPercent / 10) > (lastPercent / 10))) { /* make sure to also print out missed steps */ for (LONG curVal = (lastPercent / 10) * 10 + 10; curVal <= (currentPercent / 10) * 10; curVal += 10) { if (curVal < 100) { RTPrintf("%ld%%...", curVal); RTStrmFlush(g_pStdOut); } } } lastPercent = currentPercent; if (fCompleted) { RTPrintf("100%%\n"); RTStrmFlush(g_pStdOut); break; } /* make sure the loop is not too tight */ RTThreadSleep(100); } } static void showSnapshots(ComPtr rootSnapshot, int level = 0) { /* start with the root */ Bstr name; Guid uuid; rootSnapshot->COMGETTER(Name)(name.asOutParam()); rootSnapshot->COMGETTER(Id)(uuid.asOutParam()); /* print with indentation */ RTPrintf("%*sName: %lS (UUID: %s)\n", level * 3, "", name.raw(), uuid.toString().raw()); /* get the children */ ComPtr coll; rootSnapshot->COMGETTER(Children)(coll.asOutParam()); if (coll) { ComPtr enumerator; coll->Enumerate(enumerator.asOutParam()); BOOL hasMore = FALSE; while (enumerator->HasMore(&hasMore), hasMore) { ComPtr snapshot; enumerator->GetNext(snapshot.asOutParam()); if (snapshot) { /* recursive call */ showSnapshots(snapshot, level + 1); } } } } static void makeTimeStr (char *s, int cb, int64_t millies) { RTTIME t; RTTIMESPEC ts; RTTimeSpecSetMilli(&ts, millies); RTTimeExplode (&t, &ts); RTStrPrintf(s, cb, "%04d/%02d/%02d %02d:%02d:%02d UTC", t.i32Year, t.u8Month, t.u8MonthDay, t.u8Hour, t.u8Minute, t.u8Second); } static HRESULT showVMInfo (ComPtr virtualBox, ComPtr machine, ComPtr console = ComPtr (), bool fDetails = false) { HRESULT rc; BOOL accessible = FALSE; CHECK_ERROR (machine, COMGETTER(Accessible) (&accessible)); CheckComRCReturnRC (rc); if (!accessible) { RTPrintf ("Name: \n"); Guid uuid; rc = machine->COMGETTER(Id) (uuid.asOutParam()); RTPrintf ("UUID: %s\n", uuid.toString().raw()); Bstr settingsFilePath; rc = machine->COMGETTER(SettingsFilePath) (settingsFilePath.asOutParam()); RTPrintf ("Config file: %lS\n", settingsFilePath.raw()); ComPtr accessError; rc = machine->COMGETTER(AccessError) (accessError.asOutParam()); RTPrintf ("Access error details:\n"); ErrorInfo ei (accessError); ei.print ("[-] "); RTPrintf ("\n"); return S_OK; } Bstr machineName; rc = machine->COMGETTER(Name)(machineName.asOutParam()); RTPrintf("Name: %lS\n", machineName.raw()); ComPtr osType; Bstr osName; rc = machine->COMGETTER(OSType)(osType.asOutParam()); rc = osType->COMGETTER(Description)(osName.asOutParam()); RTPrintf("Guest OS: %lS\n", osName.raw()); Guid uuid; rc = machine->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("UUID: %s\n", uuid.toString().raw()); Bstr settingsFilePath; rc = machine->COMGETTER(SettingsFilePath)(settingsFilePath.asOutParam()); RTPrintf("Config file: %lS\n", settingsFilePath.raw()); ULONG memorySize; rc = machine->COMGETTER(MemorySize)(&memorySize); RTPrintf("Memory size: %uMB\n", memorySize); ULONG vramSize; rc = machine->COMGETTER(VRAMSize)(&vramSize); RTPrintf("VRAM size: %uMB\n", vramSize); ComPtr biosSettings; machine->COMGETTER(BIOSSettings)(biosSettings.asOutParam()); BIOSBootMenuMode_T bootMenuMode; biosSettings->COMGETTER(BootMenuMode)(&bootMenuMode); RTPrintf("Boot menu mode: "); switch (bootMenuMode) { case BIOSBootMenuMode_Disabled: RTPrintf("disabled\n"); break; case BIOSBootMenuMode_MenuOnly: RTPrintf("menu only\n"); break; default: RTPrintf("message and menu\n"); } BOOL acpiEnabled; biosSettings->COMGETTER(ACPIEnabled)(&acpiEnabled); RTPrintf("ACPI: %s\n", acpiEnabled ? "on" : "off"); BOOL ioapicEnabled; biosSettings->COMGETTER(IOAPICEnabled)(&ioapicEnabled); RTPrintf("IOAPIC: %s\n", ioapicEnabled ? "on" : "off"); TriStateBool_T hwVirtExEnabled; machine->COMGETTER(HWVirtExEnabled)(&hwVirtExEnabled); if (hwVirtExEnabled == TriStateBool_Default) { BOOL fHWVirtExEnabled; ComPtr systemProperties; virtualBox->COMGETTER(SystemProperties)(systemProperties.asOutParam()); systemProperties->COMGETTER(HWVirtExEnabled)(&fHWVirtExEnabled); RTPrintf("Hardw. virt.ext: Default (%s)\n", fHWVirtExEnabled ? "on" : "off"); } else { RTPrintf("Hardw. virt.ext: %s\n", hwVirtExEnabled == TriStateBool_True ? "on" : "off"); } MachineState_T machineState; const char *pszState = NULL; rc = machine->COMGETTER(State)(&machineState); switch (machineState) { case MachineState_PoweredOff: pszState = "powered off"; break; case MachineState_Saved: pszState = "saved"; break; case MachineState_Aborted: pszState = "aborted"; break; case MachineState_Running: pszState = "running"; break; case MachineState_Paused: pszState = "paused"; break; case MachineState_Starting: pszState = "starting"; break; case MachineState_Stopping: pszState = "stopping"; break; case MachineState_Saving: pszState = "saving"; break; case MachineState_Restoring: pszState = "restoring"; break; default: pszState = "unknown"; break; } RTPrintf("State: %s\n", pszState); ComPtr floppyDrive; rc = machine->COMGETTER(FloppyDrive)(floppyDrive.asOutParam()); if (SUCCEEDED(rc) && floppyDrive) { BOOL fFloppyEnabled; floppyDrive->COMGETTER(Enabled)(&fFloppyEnabled); Utf8Str pszFloppy = "invalid"; if (fFloppyEnabled) { DriveState_T floppyState; floppyDrive->COMGETTER(State)(&floppyState); switch (floppyState) { case DriveState_ImageMounted: { ComPtr floppyImage; rc = floppyDrive->GetImage(floppyImage.asOutParam()); if (SUCCEEDED(rc) && floppyImage) { Bstr imagePath; floppyImage->COMGETTER(FilePath)(imagePath.asOutParam()); Guid imageGuid; floppyImage->COMGETTER(Id)(imageGuid.asOutParam()); pszFloppy = Utf8StrFmt("%lS (UUID: %s)", imagePath.raw(), imageGuid.toString().raw()); } break; } case DriveState_HostDriveCaptured: { ComPtr hostFloppyDrive; rc = floppyDrive->GetHostDrive(hostFloppyDrive.asOutParam()); if (SUCCEEDED(rc) && floppyDrive) { Bstr driveName; hostFloppyDrive->COMGETTER(Name)(driveName.asOutParam()); pszFloppy = Utf8StrFmt("Host drive %lS", driveName.raw()); } break; } case DriveState_NotMounted: { pszFloppy = "empty"; break; } } } else { pszFloppy = "disabled"; } RTPrintf("Floppy: %s\n", pszFloppy.raw()); } ComPtr hardDisk; Bstr filePath; rc = machine->GetHardDisk(DiskControllerType_IDE0Controller, 0, hardDisk.asOutParam()); if (SUCCEEDED(rc) && hardDisk) { /// @todo (dmik) we temporarily use the location property to // determine the image file name. This is subject to change // when iSCSI disks are here (we should either query a // storage-specific interface from IHardDisk, or "standardize" // the location property) hardDisk->COMGETTER(Location)(filePath.asOutParam()); hardDisk->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("Primary master: %lS (UUID: %s)\n", filePath.raw(), uuid.toString().raw()); } rc = machine->GetHardDisk(DiskControllerType_IDE0Controller, 1, hardDisk.asOutParam()); if (SUCCEEDED(rc) && hardDisk) { /// @todo (dmik) we temporarily use the location property to // determine the image file name. This is subject to change // when iSCSI disks are here (we should either query a // storage-specific interface from IHardDisk, or "standardize" // the location property) hardDisk->COMGETTER(Location)(filePath.asOutParam()); hardDisk->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("Primary slave: %lS (UUID: %s)\n", filePath.raw(), uuid.toString().raw()); } rc = machine->GetHardDisk(DiskControllerType_IDE1Controller, 1, hardDisk.asOutParam()); if (SUCCEEDED(rc) && hardDisk) { /// @todo (dmik) we temporarily use the location property to // determine the image file name. This is subject to change // when iSCSI disks are here (we should either query a // storage-specific interface from IHardDisk, or "standardize" // the location property) hardDisk->COMGETTER(Location)(filePath.asOutParam()); hardDisk->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("Secondary slave: %lS (UUID: %s)\n", filePath.raw(), uuid.toString().raw()); } ComPtr dvdDrive; rc = machine->COMGETTER(DVDDrive)(dvdDrive.asOutParam()); if (SUCCEEDED(rc) && dvdDrive) { ComPtr dvdImage; rc = dvdDrive->GetImage(dvdImage.asOutParam()); if (SUCCEEDED(rc) && dvdImage) { rc = dvdImage->COMGETTER(FilePath)(filePath.asOutParam()); if (SUCCEEDED(rc) && filePath) { rc = dvdImage->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("DVD: %lS (UUID: %s)\n", filePath.raw(), uuid.toString().raw()); } } else { ComPtr hostDVDDrive; rc = dvdDrive->GetHostDrive(hostDVDDrive.asOutParam()); if (SUCCEEDED(rc) && hostDVDDrive) { Bstr name; hostDVDDrive->COMGETTER(Name)(name.asOutParam()); RTPrintf("DVD: Host drive %lS", name.raw()); } else RTPrintf("DVD: empty"); BOOL fPassthrough; dvdDrive->COMGETTER(Passthrough)(&fPassthrough); if (fPassthrough) RTPrintf(" (passthrough enabled)"); RTPrintf("\n"); } } /* get the maximum amount of NICS */ ComPtr sysProps; virtualBox->COMGETTER(SystemProperties)(sysProps.asOutParam()); ULONG maxNICs = 0; sysProps->COMGETTER(NetworkAdapterCount)(&maxNICs); for (ULONG currentNIC = 0; currentNIC < maxNICs; currentNIC++) { ComPtr nic; rc = machine->GetNetworkAdapter(currentNIC, nic.asOutParam()); if (SUCCEEDED(rc) && nic) { BOOL fEnabled; nic->COMGETTER(Enabled)(&fEnabled); if (!fEnabled) { RTPrintf("NIC %d: disabled\n", currentNIC + 1); } else { Bstr strMACAddress; nic->COMGETTER(MACAddress)(strMACAddress.asOutParam()); Utf8Str strAttachment; NetworkAttachmentType_T attachment; nic->COMGETTER(AttachmentType)(&attachment); switch (attachment) { case NetworkAttachmentType_NoNetworkAttachment: strAttachment = "none"; break; case NetworkAttachmentType_NATNetworkAttachment: strAttachment = "NAT"; break; case NetworkAttachmentType_HostInterfaceNetworkAttachment: strAttachment = "Host Interface"; break; case NetworkAttachmentType_InternalNetworkAttachment: { Bstr strNetwork; nic->COMGETTER(InternalNetwork)(strNetwork.asOutParam()); strAttachment = Utf8StrFmt("Internal Network '%s'", Utf8Str(strNetwork).raw()); break; } default: strAttachment = "unknown"; break; } /* trace stuff */ BOOL fTraceEnabled; nic->COMGETTER(TraceEnabled)(&fTraceEnabled); Bstr traceFile; nic->COMGETTER(TraceFile)(traceFile.asOutParam()); RTPrintf("NIC %d: MAC: %lS, Attachment: %s, Trace: %s (file: %lS)\n", currentNIC + 1, strMACAddress.raw(), strAttachment.raw(), fTraceEnabled ? "on" : "off", traceFile.raw()); } } } ComPtr AudioAdapter; rc = machine->COMGETTER(AudioAdapter)(AudioAdapter.asOutParam()); if (SUCCEEDED(rc)) { const char *psz = "Unknown"; BOOL fEnabled; rc = AudioAdapter->COMGETTER(Enabled)(&fEnabled); if (SUCCEEDED(rc) && fEnabled) { AudioDriverType_T enmType; rc = AudioAdapter->COMGETTER(AudioDriver)(&enmType); switch (enmType) { case AudioDriverType_NullAudioDriver: psz = "Null"; break; case AudioDriverType_WINMMAudioDriver: psz = "WINMM"; break; case AudioDriverType_DSOUNDAudioDriver: psz = "DSOUND"; break; case AudioDriverType_OSSAudioDriver: psz = "OSS"; break; case AudioDriverType_ALSAAudioDriver: psz = "ALSA"; break; case AudioDriverType_CoreAudioDriver: psz = "CoreAudio"; break; default: ; break; } } else fEnabled = FALSE; RTPrintf("Audio: %s (Driver: %s)\n", fEnabled ? "enabled" : "disabled", psz); } /* Shared clipboard */ { const char *psz = "Unknown"; ClipboardMode_T enmMode; rc = machine->COMGETTER(ClipboardMode)(&enmMode); switch (enmMode) { case ClipboardMode_ClipDisabled: psz = "Disabled"; break; case ClipboardMode_ClipHostToGuest: psz = "HostToGuest"; break; case ClipboardMode_ClipGuestToHost: psz = "GuestToHost"; break; case ClipboardMode_ClipBidirectional: psz = "Bidirectional"; break; default: ; break; } RTPrintf("Clipboard Mode: %s\n", psz); } if (console) { ComPtr display; CHECK_ERROR_RET(console, COMGETTER(Display)(display.asOutParam()), rc); ULONG xRes, yRes, bpp; CHECK_ERROR_RET(display, COMGETTER(Width)(&xRes), rc); CHECK_ERROR_RET(display, COMGETTER(Height)(&yRes), rc); CHECK_ERROR_RET(display, COMGETTER(ColorDepth)(&bpp), rc); RTPrintf("Video mode: %dx%dx%d\n", xRes, yRes, bpp); } /* * VRDP */ ComPtr vrdpServer; rc = machine->COMGETTER(VRDPServer)(vrdpServer.asOutParam()); if (SUCCEEDED(rc) && vrdpServer) { BOOL fEnabled = false; vrdpServer->COMGETTER(Enabled)(&fEnabled); if (fEnabled) { ULONG port; vrdpServer->COMGETTER(Port)(&port); VRDPAuthType_T vrdpAuthType; const char *strAuthType; vrdpServer->COMGETTER(AuthType)(&vrdpAuthType); switch (vrdpAuthType) { case VRDPAuthType_VRDPAuthNull: strAuthType = "null"; break; case VRDPAuthType_VRDPAuthExternal: strAuthType = "external"; break; case VRDPAuthType_VRDPAuthGuest: strAuthType = "guest"; break; default: strAuthType = "unknown"; break; } RTPrintf("VRDP: enabled (Port %d, Authentication type: %s)\n", port, strAuthType); } else RTPrintf("VRDP: disabled\n"); } /* * USB. */ ComPtr USBCtl; rc = machine->COMGETTER(USBController)(USBCtl.asOutParam()); if (SUCCEEDED(rc)) { BOOL fEnabled; rc = USBCtl->COMGETTER(Enabled)(&fEnabled); if (FAILED(rc)) fEnabled = false; RTPrintf("USB: %s\n", fEnabled ? "enabled" : "disabled"); RTPrintf("\nUSB Device Filters:\n\n"); ComPtr Coll; CHECK_ERROR_RET (USBCtl, COMGETTER(DeviceFilters)(Coll.asOutParam()), rc); ComPtr Enum; CHECK_ERROR_RET (Coll, Enumerate(Enum.asOutParam()), rc); ULONG index = 0; BOOL fMore = FALSE; rc = Enum->HasMore (&fMore); ASSERT_RET (SUCCEEDED (rc), rc); if (!fMore) { RTPrintf("\n\n"); } else while (fMore) { ComPtr DevPtr; rc = Enum->GetNext(DevPtr.asOutParam()); ASSERT_RET (SUCCEEDED (rc), rc); /* Query info. */ RTPrintf("Index: %lu\n", index); BOOL bActive = FALSE; CHECK_ERROR_RET (DevPtr, COMGETTER (Active) (&bActive), rc); RTPrintf("Active: %s\n", bActive ? "yes" : "no"); Bstr bstr; CHECK_ERROR_RET (DevPtr, COMGETTER (Name) (bstr.asOutParam()), rc); RTPrintf("Name: %lS\n", bstr.raw()); CHECK_ERROR_RET (DevPtr, COMGETTER (VendorId) (bstr.asOutParam()), rc); RTPrintf("VendorId: %lS\n", bstr.raw()); CHECK_ERROR_RET (DevPtr, COMGETTER (ProductId) (bstr.asOutParam()), rc); RTPrintf("ProductId: %lS\n", bstr.raw()); CHECK_ERROR_RET (DevPtr, COMGETTER (Revision) (bstr.asOutParam()), rc); RTPrintf("Revision: %lS\n", bstr.raw()); CHECK_ERROR_RET (DevPtr, COMGETTER (Manufacturer) (bstr.asOutParam()), rc); RTPrintf("Manufacturer: %lS\n", bstr.raw()); CHECK_ERROR_RET (DevPtr, COMGETTER (Product) (bstr.asOutParam()), rc); RTPrintf("Product: %lS\n", bstr.raw()); CHECK_ERROR_RET (DevPtr, COMGETTER (SerialNumber) (bstr.asOutParam()), rc); RTPrintf("Serial Number: %lS\n\n", bstr.raw()); rc = Enum->HasMore (&fMore); ASSERT_RET (SUCCEEDED (rc), rc); index ++; } if (console) { /* scope */ { RTPrintf("Available remote USB devices:\n\n"); ComPtr coll; CHECK_ERROR_RET (console, COMGETTER(RemoteUSBDevices) (coll.asOutParam()), rc); ComPtr en; CHECK_ERROR_RET (coll, Enumerate (en.asOutParam()), rc); BOOL more = FALSE; rc = en->HasMore (&more); ASSERT_RET (SUCCEEDED (rc), rc); if (!more) { RTPrintf("\n\n"); } else while (more) { ComPtr dev; rc = en->GetNext (dev.asOutParam()); ASSERT_RET (SUCCEEDED (rc), rc); /* Query info. */ Guid id; CHECK_ERROR_RET (dev, COMGETTER(Id)(id.asOutParam()), rc); USHORT usVendorId; CHECK_ERROR_RET (dev, COMGETTER(VendorId)(&usVendorId), rc); USHORT usProductId; CHECK_ERROR_RET (dev, COMGETTER(ProductId)(&usProductId), rc); USHORT bcdRevision; CHECK_ERROR_RET (dev, COMGETTER(Revision)(&bcdRevision), rc); RTPrintf("UUID: %S\n" "VendorId: 0x%04x (%04X)\n" "ProductId: 0x%04x (%04X)\n" "Revision: %u.%u (%02u%02u)\n", id.toString().raw(), usVendorId, usVendorId, usProductId, usProductId, bcdRevision >> 8, bcdRevision & 0xff, bcdRevision >> 8, bcdRevision & 0xff); /* optional stuff. */ Bstr bstr; CHECK_ERROR_RET (dev, COMGETTER(Manufacturer)(bstr.asOutParam()), rc); if (!bstr.isEmpty()) RTPrintf("Manufacturer: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(Product)(bstr.asOutParam()), rc); if (!bstr.isEmpty()) RTPrintf("Product: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(SerialNumber)(bstr.asOutParam()), rc); if (!bstr.isEmpty()) RTPrintf("SerialNumber: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(Address)(bstr.asOutParam()), rc); if (!bstr.isEmpty()) RTPrintf("Address: %lS\n", bstr.raw()); RTPrintf("\n"); rc = en->HasMore (&more); ASSERT_RET (SUCCEEDED (rc), rc); } } /* scope */ { RTPrintf ("Currently Attached USB Devices:\n\n"); ComPtr coll; CHECK_ERROR_RET (console, COMGETTER(USBDevices) (coll.asOutParam()), rc); ComPtr en; CHECK_ERROR_RET (coll, Enumerate (en.asOutParam()), rc); BOOL more = FALSE; rc = en->HasMore (&more); ASSERT_RET (SUCCEEDED (rc), rc); if (!more) { RTPrintf("\n\n"); } else while (more) { ComPtr dev; rc = en->GetNext (dev.asOutParam()); ASSERT_RET (SUCCEEDED (rc), rc); /* Query info. */ Guid id; CHECK_ERROR_RET (dev, COMGETTER(Id)(id.asOutParam()), rc); USHORT usVendorId; CHECK_ERROR_RET (dev, COMGETTER(VendorId)(&usVendorId), rc); USHORT usProductId; CHECK_ERROR_RET (dev, COMGETTER(ProductId)(&usProductId), rc); USHORT bcdRevision; CHECK_ERROR_RET (dev, COMGETTER(Revision)(&bcdRevision), rc); RTPrintf("UUID: %S\n" "VendorId: 0x%04x (%04X)\n" "ProductId: 0x%04x (%04X)\n" "Revision: %u.%u (%02u%02u)\n", id.toString().raw(), usVendorId, usVendorId, usProductId, usProductId, bcdRevision >> 8, bcdRevision & 0xff, bcdRevision >> 8, bcdRevision & 0xff); /* optional stuff. */ Bstr bstr; CHECK_ERROR_RET (dev, COMGETTER(Manufacturer)(bstr.asOutParam()), rc); if (!bstr.isEmpty()) RTPrintf("Manufacturer: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(Product)(bstr.asOutParam()), rc); if (!bstr.isEmpty()) RTPrintf("Product: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(SerialNumber)(bstr.asOutParam()), rc); if (!bstr.isEmpty()) RTPrintf("SerialNumber: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(Address)(bstr.asOutParam()), rc); if (!bstr.isEmpty()) RTPrintf("Address: %lS\n", bstr.raw()); RTPrintf("\n"); rc = en->HasMore (&more); ASSERT_RET (SUCCEEDED (rc), rc); } } } } /* USB */ /* * Shared folders */ RTPrintf("Shared folders:\n\n"); uint32_t numSharedFolders = 0; #if 0 // not yet implemented /* globally shared folders first */ { ComPtr sfColl; ComPtr sfEnum; CHECK_ERROR_RET(virtualBox, COMGETTER(SharedFolders)(sfColl.asOutParam()), rc); CHECK_ERROR_RET(sfColl, Enumerate(sfEnum.asOutParam()), rc); BOOL fMore; sfEnum->HasMore(&fMore); while (fMore) { ComPtr sf; CHECK_ERROR_RET(sfEnum, GetNext(sf.asOutParam()), rc); Bstr name, hostPath; sf->COMGETTER(Name)(name.asOutParam()); sf->COMGETTER(HostPath)(hostPath.asOutParam()); RTPrintf("Name: '%lS', Host path: '%lS' (global mapping)\n", name.raw(), hostPath.raw()); ++numSharedFolders; CHECK_ERROR_RET(sfEnum, HasMore(&fMore), rc); } } #endif /* now VM mappings */ { ComPtr sfColl; ComPtr sfEnum; CHECK_ERROR_RET(machine, COMGETTER(SharedFolders)(sfColl.asOutParam()), rc); CHECK_ERROR_RET(sfColl, Enumerate(sfEnum.asOutParam()), rc); BOOL fMore; sfEnum->HasMore(&fMore); while (fMore) { ComPtr sf; CHECK_ERROR_RET(sfEnum, GetNext(sf.asOutParam()), rc); Bstr name, hostPath; sf->COMGETTER(Name)(name.asOutParam()); sf->COMGETTER(HostPath)(hostPath.asOutParam()); RTPrintf("Name: '%lS', Host path: '%lS' (machine mapping)\n", name.raw(), hostPath.raw()); ++numSharedFolders; CHECK_ERROR_RET(sfEnum, HasMore(&fMore), rc); } } /* transient mappings */ if (console) { ComPtr sfColl; ComPtr sfEnum; CHECK_ERROR_RET(console, COMGETTER(SharedFolders)(sfColl.asOutParam()), rc); CHECK_ERROR_RET(sfColl, Enumerate(sfEnum.asOutParam()), rc); BOOL fMore; sfEnum->HasMore(&fMore); while (fMore) { ComPtr sf; CHECK_ERROR_RET(sfEnum, GetNext(sf.asOutParam()), rc); Bstr name, hostPath; sf->COMGETTER(Name)(name.asOutParam()); sf->COMGETTER(HostPath)(hostPath.asOutParam()); RTPrintf("Name: '%lS', Host path: '%lS' (transient mapping)\n", name.raw(), hostPath.raw()); ++numSharedFolders; CHECK_ERROR_RET(sfEnum, HasMore(&fMore), rc); } } if (!numSharedFolders) RTPrintf("\n"); RTPrintf("\n"); if (console) { /* * Live VRDP info. */ ComPtr remoteDisplayInfo; CHECK_ERROR_RET(console, COMGETTER(RemoteDisplayInfo)(remoteDisplayInfo.asOutParam()), rc); BOOL Active; ULONG NumberOfClients; LONG64 BeginTime; LONG64 EndTime; ULONG64 BytesSent; ULONG64 BytesSentTotal; ULONG64 BytesReceived; ULONG64 BytesReceivedTotal; Bstr User; Bstr Domain; Bstr ClientName; Bstr ClientIP; ULONG ClientVersion; ULONG EncryptionStyle; CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(Active) (&Active), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(NumberOfClients) (&NumberOfClients), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(BeginTime) (&BeginTime), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(EndTime) (&EndTime), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(BytesSent) (&BytesSent), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(BytesSentTotal) (&BytesSentTotal), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(BytesReceived) (&BytesReceived), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(BytesReceivedTotal) (&BytesReceivedTotal), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(User) (User.asOutParam ()), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(Domain) (Domain.asOutParam ()), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(ClientName) (ClientName.asOutParam ()), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(ClientIP) (ClientIP.asOutParam ()), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(ClientVersion) (&ClientVersion), rc); CHECK_ERROR_RET(remoteDisplayInfo, COMGETTER(EncryptionStyle) (&EncryptionStyle), rc); RTPrintf("VRDP Connection: %s\n", Active? "active": "not active"); RTPrintf("Clients so far: %d\n", NumberOfClients); if (NumberOfClients > 0) { char timestr[128]; if (Active) { makeTimeStr (timestr, sizeof (timestr), BeginTime); RTPrintf("Start time: %s\n", timestr); } else { makeTimeStr (timestr, sizeof (timestr), BeginTime); RTPrintf("Last started: %s\n", timestr); makeTimeStr (timestr, sizeof (timestr), EndTime); RTPrintf("Last ended: %s\n", timestr); } RTPrintf("Sent: %llu Bytes\n", BytesSent); RTPrintf("Average speed: %llu B/s\n", (BytesSent * 1000) / (EndTime - BeginTime) ); RTPrintf("Sent total: %llu Bytes\n", BytesSentTotal); RTPrintf("Received: %llu Bytes\n", BytesReceived); RTPrintf("Speed: %llu B/s\n", (BytesReceived * 1000) / (EndTime - BeginTime) ); RTPrintf("Received total: %llu Bytes\n", BytesReceivedTotal); if (Active) { RTPrintf("User name: %lS\n", User.raw()); RTPrintf("Domain: %lS\n", Domain.raw()); RTPrintf("Client name: %lS\n", ClientName.raw()); RTPrintf("Client IP: %lS\n", ClientIP.raw()); RTPrintf("Client version: %d\n", ClientVersion); RTPrintf("Encryption: %s\n", EncryptionStyle == 0? "RDP4": "RDP5 (X.509)"); } } RTPrintf("\n"); } if (fDetails) { Bstr description; machine->COMGETTER(Description)(description.asOutParam()); if (!description.isEmpty()) { RTPrintf("Description:\n%lS\n", description.raw()); } } /* * snapshots */ ComPtr snapshot; rc = machine->GetSnapshot(Guid(), snapshot.asOutParam()); if (SUCCEEDED(rc) && snapshot) { RTPrintf("Snapshots:\n\n"); showSnapshots(snapshot); } RTPrintf("\n"); return S_OK; } static int handleShowVMInfo(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; /* at least one option: the UUID or name of the VM */ if (argc < 1) { return errorSyntax(USAGE_SHOWVMINFO, "Incorrect number of parameters"); } /* try to find the given machine */ ComPtr machine; Guid uuid (argv[0]); if (!uuid.isEmpty()) { CHECK_ERROR (virtualBox, GetMachine (uuid, machine.asOutParam())); } else { CHECK_ERROR (virtualBox, FindMachine (Bstr(argv[0]), machine.asOutParam())); if (SUCCEEDED (rc)) machine->COMGETTER(Id) (uuid.asOutParam()); } if (FAILED (rc)) return 1; /* 2nd option can be -details */ bool fDetails = false; if ((argc == 2) && !strcmp(argv[1], "-details")) fDetails = true; ComPtr console; /* open an existing session for the VM */ rc = virtualBox->OpenExistingSession (session, uuid); if (SUCCEEDED(rc)) /* get the session machine */ rc = session->COMGETTER(Machine)(machine.asOutParam()); if (SUCCEEDED(rc)) /* get the session console */ rc = session->COMGETTER(Console)(console.asOutParam()); rc = showVMInfo (virtualBox, machine, console, fDetails); if (console) session->Close(); return SUCCEEDED (rc) ? 0 : 1; } static int handleList(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc = S_OK; /* exactly one option: the object */ if (argc != 1) { return errorSyntax(USAGE_LIST, "Incorrect number of parameters"); } /* which object? */ if (strcmp(argv[0], "vms") == 0) { /* * Get the list of all registered VMs */ ComPtr collection; rc = virtualBox->COMGETTER(Machines)(collection.asOutParam()); ComPtr enumerator; if (SUCCEEDED(rc)) rc = collection->Enumerate(enumerator.asOutParam()); if (SUCCEEDED(rc)) { /* * Iterate through the collection */ BOOL hasMore = FALSE; while (enumerator->HasMore(&hasMore), hasMore) { ComPtr machine; rc = enumerator->GetNext(machine.asOutParam()); if ((SUCCEEDED(rc)) && machine) { rc = showVMInfo(virtualBox, machine); } } } } else if (strcmp(argv[0], "ostypes") == 0) { ComPtr coll; ComPtr enumerator; CHECK_ERROR(virtualBox, COMGETTER(GuestOSTypes)(coll.asOutParam())); if (SUCCEEDED(rc) && coll) { CHECK_ERROR(coll, Enumerate(enumerator.asOutParam())); BOOL hasMore; while (SUCCEEDED(enumerator->HasMore(&hasMore)) && hasMore) { ComPtr guestOS; CHECK_RC_BREAK(enumerator->GetNext(guestOS.asOutParam())); Bstr guestId; guestOS->COMGETTER(Id)(guestId.asOutParam()); RTPrintf("ID: %lS\n", guestId.raw()); Bstr guestDescription; guestOS->COMGETTER(Description)(guestDescription.asOutParam()); RTPrintf("Description: %lS\n\n", guestDescription.raw()); } } } else if (strcmp(argv[0], "hostdvds") == 0) { ComPtr host; CHECK_ERROR(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr coll; ComPtr enumerator; CHECK_ERROR(host, COMGETTER(DVDDrives)(coll.asOutParam())); if (SUCCEEDED(rc) && coll) { CHECK_ERROR(coll, Enumerate(enumerator.asOutParam())); BOOL hasMore; while (SUCCEEDED(enumerator->HasMore(&hasMore)) && hasMore) { ComPtr dvdDrive; CHECK_RC_BREAK(enumerator->GetNext(dvdDrive.asOutParam())); Bstr name; dvdDrive->COMGETTER(Name)(name.asOutParam()); RTPrintf("Name: %lS\n\n", name.raw()); } } } else if (strcmp(argv[0], "hostfloppies") == 0) { ComPtr host; CHECK_ERROR(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr coll; ComPtr enumerator; CHECK_ERROR(host, COMGETTER(FloppyDrives)(coll.asOutParam())); if (SUCCEEDED(rc) && coll) { CHECK_ERROR(coll, Enumerate(enumerator.asOutParam())); BOOL hasMore; while (SUCCEEDED(enumerator->HasMore(&hasMore)) && hasMore) { ComPtr floppyDrive; CHECK_RC_BREAK(enumerator->GetNext(floppyDrive.asOutParam())); Bstr name; floppyDrive->COMGETTER(Name)(name.asOutParam()); RTPrintf("Name: %lS\n\n", name.raw()); } } } #ifdef __WIN__ else if (strcmp(argv[0], "hostifs") == 0) { ComPtr host; CHECK_ERROR(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr coll; ComPtr enumerator; CHECK_ERROR(host, COMGETTER(NetworkInterfaces)(coll.asOutParam())); if (SUCCEEDED(rc) && coll) { CHECK_ERROR(coll, Enumerate(enumerator.asOutParam())); BOOL hasMore; while (SUCCEEDED(enumerator->HasMore(&hasMore)) && hasMore) { ComPtr networkInterface; CHECK_RC_BREAK(enumerator->GetNext(networkInterface.asOutParam())); Bstr interfaceName; networkInterface->COMGETTER(Name)(interfaceName.asOutParam()); RTPrintf("Name: %lS\n", interfaceName.raw()); Guid interfaceGuid; networkInterface->COMGETTER(Id)(interfaceGuid.asOutParam()); RTPrintf("GUID: %lS\n\n", Bstr(interfaceGuid.toString())); } } } #endif /* __WIN__ */ else if (strcmp(argv[0], "hdds") == 0) { ComPtr hddColl; CHECK_ERROR(virtualBox, COMGETTER(HardDisks)(hddColl.asOutParam())); ComPtr enumerator; CHECK_ERROR(hddColl, Enumerate(enumerator.asOutParam())); BOOL hasMore; while (SUCCEEDED(enumerator->HasMore(&hasMore)) && hasMore) { ComPtr hdd; CHECK_RC_BREAK(enumerator->GetNext(hdd.asOutParam())); Guid uuid; hdd->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("UUID: %s\n", uuid.toString().raw()); HardDiskStorageType_T storageType; hdd->COMGETTER(StorageType)(&storageType); const char *storageTypeString = "unknown"; switch (storageType) { case HardDiskStorageType_VirtualDiskImage: storageTypeString = "Virtual Disk Image"; break; case HardDiskStorageType_ISCSIHardDisk: storageTypeString = "iSCSI hard disk"; break; } RTPrintf("Storage type: %s\n", storageTypeString); Bstr filepath; /// @todo (dmik) we temporarily use the location property to // determine the image file name. This is subject to change // when iSCSI disks are here (we should either query a // storage-specific interface from IHardDisk, or "standardize" // the location property) hdd->COMGETTER(Location)(filepath.asOutParam()); RTPrintf("Path: %lS\n", filepath.raw()); BOOL fAccessible; hdd->COMGETTER(AllAccessible)(&fAccessible); RTPrintf("Accessible: %s\n", fAccessible ? "yes" : "no"); Guid machineUUID; hdd->COMGETTER(MachineId)(machineUUID.asOutParam()); if (!machineUUID.isEmpty()) { ComPtr machine; CHECK_ERROR(virtualBox, GetMachine(machineUUID, machine.asOutParam())); ASSERT(machine); Bstr name; machine->COMGETTER(Name)(name.asOutParam()); machine->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("Usage: %lS (UUID: %s)\n", name.raw(), uuid.toString().raw()); } RTPrintf("\n"); } } else if (strcmp(argv[0], "dvds") == 0) { ComPtr dvdColl; CHECK_ERROR(virtualBox, COMGETTER(DVDImages)(dvdColl.asOutParam())); ComPtr enumerator; CHECK_ERROR(dvdColl, Enumerate(enumerator.asOutParam())); BOOL hasMore; while (SUCCEEDED(enumerator->HasMore(&hasMore)) && hasMore) { ComPtr dvdImage; CHECK_RC_BREAK(enumerator->GetNext(dvdImage.asOutParam())); Guid uuid; dvdImage->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("UUID: %s\n", uuid.toString().raw()); Bstr filePath; dvdImage->COMGETTER(FilePath)(filePath.asOutParam()); RTPrintf("Path: %lS\n", filePath.raw()); BOOL fAccessible; dvdImage->COMGETTER(Accessible)(&fAccessible); RTPrintf("Accessible: %s\n", fAccessible ? "yes" : "no"); Bstr machineUUIDs; CHECK_ERROR(virtualBox, GetDVDImageUsage(uuid, ResourceUsage_AllUsage, machineUUIDs.asOutParam())); /** @todo usage */ RTPrintf("\n"); } } else if (strcmp(argv[0], "floppies") == 0) { ComPtr floppyColl; CHECK_ERROR(virtualBox, COMGETTER(FloppyImages)(floppyColl.asOutParam())); ComPtr enumerator; CHECK_ERROR(floppyColl, Enumerate(enumerator.asOutParam())); BOOL hasMore; while (SUCCEEDED(enumerator->HasMore(&hasMore)) && hasMore) { ComPtr floppyImage; CHECK_RC_BREAK(enumerator->GetNext(floppyImage.asOutParam())); Guid uuid; floppyImage->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("UUID: %s\n", uuid.toString().raw()); Bstr filePath; floppyImage->COMGETTER(FilePath)(filePath.asOutParam()); RTPrintf("Path: %lS\n", filePath.raw()); BOOL fAccessible; floppyImage->COMGETTER(Accessible)(&fAccessible); RTPrintf("Accessible: %s\n", fAccessible ? "yes" : "no"); Bstr machineUUIDs; CHECK_ERROR(virtualBox, GetFloppyImageUsage(uuid, ResourceUsage_AllUsage, machineUUIDs.asOutParam())); /** @todo usage */ RTPrintf("\n"); } } else if (strcmp(argv[0], "usbhost") == 0) { ComPtr Host; CHECK_ERROR_RET (virtualBox, COMGETTER(Host)(Host.asOutParam()), 1); ComPtr CollPtr; CHECK_ERROR_RET (Host, COMGETTER(USBDevices)(CollPtr.asOutParam()), 1); ComPtr EnumPtr; CHECK_ERROR_RET (CollPtr, Enumerate(EnumPtr.asOutParam()), 1); RTPrintf("Host USB Devices:\n\n"); BOOL fMore = FALSE; rc = EnumPtr->HasMore (&fMore); ASSERT_RET (SUCCEEDED (rc), 1); if (!fMore) { RTPrintf("\n\n"); } else while (fMore) { ComPtr dev; rc = EnumPtr->GetNext (dev.asOutParam()); ASSERT_RET (SUCCEEDED (rc), 1); /* Query info. */ Guid id; CHECK_ERROR_RET (dev, COMGETTER(Id)(id.asOutParam()), 1); USHORT usVendorId; CHECK_ERROR_RET (dev, COMGETTER(VendorId)(&usVendorId), 1); USHORT usProductId; CHECK_ERROR_RET (dev, COMGETTER(ProductId)(&usProductId), 1); USHORT bcdRevision; CHECK_ERROR_RET (dev, COMGETTER(Revision)(&bcdRevision), 1); RTPrintf("UUID: %S\n" "VendorId: 0x%04x (%04X)\n" "ProductId: 0x%04x (%04X)\n" "Revision: %u.%u (%02u%02u)\n", id.toString().raw(), usVendorId, usVendorId, usProductId, usProductId, bcdRevision >> 8, bcdRevision & 0xff, bcdRevision >> 8, bcdRevision & 0xff); /* optional stuff. */ Bstr bstr; CHECK_ERROR_RET (dev, COMGETTER(Manufacturer)(bstr.asOutParam()), 1); if (!bstr.isEmpty()) RTPrintf("Manufacturer: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(Product)(bstr.asOutParam()), 1); if (!bstr.isEmpty()) RTPrintf("Product: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(SerialNumber)(bstr.asOutParam()), 1); if (!bstr.isEmpty()) RTPrintf("SerialNumber: %lS\n", bstr.raw()); CHECK_ERROR_RET (dev, COMGETTER(Address)(bstr.asOutParam()), 1); if (!bstr.isEmpty()) RTPrintf("Address: %lS\n", bstr.raw()); /* current state */ USBDeviceState_T state; CHECK_ERROR_RET (dev, COMGETTER(State)(&state), 1); const char *pszState = "?"; switch (state) { case USBDeviceState_USBDeviceNotSupported: pszState = "Not supported"; break; case USBDeviceState_USBDeviceUnavailable: pszState = "Unavailable"; break; case USBDeviceState_USBDeviceBusy: pszState = "Busy"; break; case USBDeviceState_USBDeviceAvailable: pszState = "Available"; break; case USBDeviceState_USBDeviceHeld: pszState = "Held"; break; case USBDeviceState_USBDeviceCaptured: pszState = "Captured"; break; default: ASSERT (false); break; } RTPrintf("Current State: %s\n\n", pszState); rc = EnumPtr->HasMore (&fMore); ASSERT_RET (SUCCEEDED (rc), rc); } } else if (strcmp(argv[0], "usbfilters") == 0) { RTPrintf("Global USB Device Filters:\n\n"); ComPtr host; CHECK_ERROR_RET (virtualBox, COMGETTER(Host) (host.asOutParam()), 1); ComPtr coll; CHECK_ERROR_RET (host, COMGETTER (USBDeviceFilters)(coll.asOutParam()), 1); ComPtr en; CHECK_ERROR_RET (coll, Enumerate(en.asOutParam()), 1); ULONG index = 0; BOOL more = FALSE; rc = en->HasMore (&more); ASSERT_RET (SUCCEEDED (rc), 1); if (!more) { RTPrintf("\n\n"); } else while (more) { ComPtr flt; rc = en->GetNext (flt.asOutParam()); ASSERT_RET (SUCCEEDED (rc), 1); /* Query info. */ RTPrintf("Index: %lu\n", index); BOOL active = FALSE; CHECK_ERROR_RET (flt, COMGETTER (Active) (&active), 1); RTPrintf("Active: %s\n", active ? "yes" : "no"); USBDeviceFilterAction_T action; CHECK_ERROR_RET (flt, COMGETTER (Action) (&action), 1); const char *pszAction = ""; switch (action) { case USBDeviceFilterAction_USBDeviceFilterIgnore: pszAction = "Ignore"; break; case USBDeviceFilterAction_USBDeviceFilterHold: pszAction = "Hold"; break; default: break; } RTPrintf("Action: %s\n", pszAction); Bstr bstr; CHECK_ERROR_RET (flt, COMGETTER (Name) (bstr.asOutParam()), 1); RTPrintf("Name: %lS\n", bstr.raw()); CHECK_ERROR_RET (flt, COMGETTER (VendorId) (bstr.asOutParam()), 1); RTPrintf("VendorId: %lS\n", bstr.raw()); CHECK_ERROR_RET (flt, COMGETTER (ProductId) (bstr.asOutParam()), 1); RTPrintf("ProductId: %lS\n", bstr.raw()); CHECK_ERROR_RET (flt, COMGETTER (Revision) (bstr.asOutParam()), 1); RTPrintf("Revision: %lS\n", bstr.raw()); CHECK_ERROR_RET (flt, COMGETTER (Manufacturer) (bstr.asOutParam()), 1); RTPrintf("Manufacturer: %lS\n", bstr.raw()); CHECK_ERROR_RET (flt, COMGETTER (Product) (bstr.asOutParam()), 1); RTPrintf("Product: %lS\n", bstr.raw()); CHECK_ERROR_RET (flt, COMGETTER (SerialNumber) (bstr.asOutParam()), 1); RTPrintf("Serial Number: %lS\n\n", bstr.raw()); rc = en->HasMore (&more); ASSERT_RET (SUCCEEDED (rc), 1); index ++; } } else if (strcmp(argv[0], "systemproperties") == 0) { ComPtr systemProperties; virtualBox->COMGETTER(SystemProperties)(systemProperties.asOutParam()); Bstr str; ULONG ulValue; ULONG64 ul64Value; BOOL flag; systemProperties->COMGETTER(MinGuestRAM)(&ulValue); RTPrintf("Minimum guest RAM size: %u Megabytes\n", ulValue); systemProperties->COMGETTER(MaxGuestRAM)(&ulValue); RTPrintf("Maximum guest RAM size: %u Megabytes\n", ulValue); systemProperties->COMGETTER(MaxGuestVRAM)(&ulValue); RTPrintf("Maximum video RAM size: %u Megabytes\n", ulValue); systemProperties->COMGETTER(MaxVDISize)(&ul64Value); RTPrintf("Maximum VDI size: %lu Megabytes\n", ul64Value); systemProperties->COMGETTER(DefaultVDIFolder)(str.asOutParam()); RTPrintf("Default VDI filder: %lS\n", str.raw()); systemProperties->COMGETTER(DefaultMachineFolder)(str.asOutParam()); RTPrintf("Default machine folder: %lS\n", str.raw()); systemProperties->COMGETTER(RemoteDisplayAuthLibrary)(str.asOutParam()); RTPrintf("VRDP authentication library: %lS\n", str.raw()); systemProperties->COMGETTER(HWVirtExEnabled)(&flag); RTPrintf("Hardware virt. extensions: %s\n", flag ? "yes" : "no"); } else { return errorSyntax(USAGE_LIST, "Invalid parameter '%s'", Utf8Str(argv[0]).raw()); } return SUCCEEDED(rc) ? 0 : 1; } static int handleRegisterVM(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; if (argc != 1) { return errorSyntax(USAGE_REGISTERVM, "Incorrect number of parameters"); } ComPtr machine; CHECK_ERROR(virtualBox, OpenMachine(Bstr(argv[0]), machine.asOutParam())); if (SUCCEEDED(rc)) { ASSERT(machine); CHECK_ERROR(virtualBox, RegisterMachine(machine)); } return SUCCEEDED(rc) ? 0 : 1; } static int handleUnregisterVM(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; if ((argc != 1) && (argc != 2)) { return errorSyntax(USAGE_UNREGISTERVM, "Incorrect number of parameters"); } ComPtr machine; /* assume it's a UUID */ rc = virtualBox->GetMachine(Guid(argv[0]), machine.asOutParam()); if (FAILED(rc) || !machine) { /* must be a name */ CHECK_ERROR(virtualBox, FindMachine(Bstr(argv[0]), machine.asOutParam())); } if (machine) { Guid uuid; machine->COMGETTER(Id)(uuid.asOutParam()); machine = NULL; CHECK_ERROR(virtualBox, UnregisterMachine(uuid, machine.asOutParam())); if (SUCCEEDED(rc) && machine) { /* are we supposed to delete the config file? */ if ((argc == 2) && (strcmp(argv[1], "-delete") == 0)) { CHECK_ERROR(machine, DeleteSettings()); } } } return SUCCEEDED(rc) ? 0 : 1; } static int handleCreateVDI(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; Bstr filename; uint64_t sizeMB = 0; bool fStatic = false; Bstr comment; bool fRegister = false; const char *type = "normal"; /* let's have a closer look at the arguments */ for (int i = 0; i < argc; i++) { if (strcmp(argv[i], "-filename") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; filename = argv[i]; } else if (strcmp(argv[i], "-size") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; sizeMB = RTStrToUInt64(argv[i]); } else if (strcmp(argv[i], "-static") == 0) { fStatic = true; } else if (strcmp(argv[i], "-comment") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; comment = argv[i]; } else if (strcmp(argv[i], "-register") == 0) { fRegister = true; } else if (strcmp(argv[i], "-type") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; type = argv[i]; } else { return errorSyntax(USAGE_CREATEVDI, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); } } /* check the outcome */ if (!filename || (sizeMB == 0)) { return errorSyntax(USAGE_CREATEVDI, "Parameters -filename and -size are required"); } if (strcmp(type, "normal") && strcmp(type, "writethrough")) { return errorArgument("Invalid VDI type '%s' specified", Utf8Str(type).raw()); } ComPtr hardDisk; CHECK_ERROR(virtualBox, CreateHardDisk(HardDiskStorageType_VirtualDiskImage, hardDisk.asOutParam())); if (SUCCEEDED(rc) && hardDisk) { CHECK_ERROR(hardDisk,COMSETTER(Description)(comment)); ComPtr vdi = hardDisk; CHECK_ERROR(vdi, COMSETTER(FilePath)(filename)); ComPtr progress; if (fStatic) { CHECK_ERROR(vdi, CreateFixedImage(sizeMB, progress.asOutParam())); } else { CHECK_ERROR(vdi, CreateDynamicImage(sizeMB, progress.asOutParam())); } if (SUCCEEDED(rc) && progress) { CHECK_ERROR(progress, WaitForCompletion(-1)); if (SUCCEEDED(rc)) { progress->COMGETTER(ResultCode)(&rc); if (FAILED(rc)) { com::ProgressErrorInfo info(progress); if (info.isBasicAvailable()) { RTPrintf("Error: failed to create disk image. Error message: %lS\n", info.getText().raw()); } else { RTPrintf("Error: failed to create disk image. No error message available!\n"); } } else { Guid uuid; CHECK_ERROR(hardDisk, COMGETTER(Id)(uuid.asOutParam())); if (strcmp(type, "normal") == 0) { /* nothing required, default */ } else if (strcmp(type, "writethrough") == 0) { CHECK_ERROR(hardDisk, COMSETTER(Type)(HardDiskType_WritethroughHardDisk)); } RTPrintf("Disk image created. UUID: %s\n", uuid.toString().raw()); } } } if (SUCCEEDED(rc) && fRegister) { CHECK_ERROR(virtualBox, RegisterHardDisk(hardDisk)); } } return SUCCEEDED(rc) ? 0 : 1; } static DECLCALLBACK(int) vdiProgressCallback(PVM pVM, unsigned uPercent, void *pvUser) { unsigned *pPercent = (unsigned *)pvUser; if (*pPercent != uPercent) { *pPercent = uPercent; RTPrintf("."); if ((uPercent % 10) == 0 && uPercent) RTPrintf("%d%%", uPercent); RTStrmFlush(g_pStdOut); } return VINF_SUCCESS; } static int handleModifyVDI(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; /* The uuid/filename and a command */ if (argc < 2) { return errorSyntax(USAGE_MODIFYVDI, "Incorrect number of parameters"); } ComPtr hardDisk; ComPtr vdi; Bstr filepath; /* first guess is that it's a UUID */ Guid uuid(argv[0]); rc = virtualBox->GetHardDisk(uuid, hardDisk.asOutParam()); /* no? then it must be a filename */ if (!hardDisk) { filepath = argv[0]; CHECK_ERROR(virtualBox, FindVirtualDiskImage(filepath, vdi.asOutParam())); hardDisk = vdi; } else { vdi = hardDisk; } /* let's find out which command */ // doesn't currently work if (strcmp(argv[1], "settype") == 0) if (0) { /* hard disk must be registered */ if (SUCCEEDED(rc) && hardDisk && vdi) { char *type = NULL; if (argc <= 2) { return errorArgument("Missing argument to for settype"); } type = argv[2]; HardDiskType_T hddType; CHECK_ERROR(hardDisk, COMGETTER(Type)(&hddType)); if (strcmp(type, "normal") == 0) { if (hddType != HardDiskType_NormalHardDisk) CHECK_ERROR(hardDisk, COMSETTER(Type)(HardDiskType_NormalHardDisk)); } else if (strcmp(type, "writethrough") == 0) { if (hddType != HardDiskType_WritethroughHardDisk) CHECK_ERROR(hardDisk, COMSETTER(Type)(HardDiskType_WritethroughHardDisk)); } else if (strcmp(type, "immutable") == 0) { if (hddType != HardDiskType_ImmutableHardDisk) CHECK_ERROR(hardDisk, COMSETTER(Type)(HardDiskType_ImmutableHardDisk)); } else { return errorArgument("Invalid VDI type '%s' specified", Utf8Str(type).raw()); } } else { return errorArgument("Hard disk image not registered"); } } else if (strcmp(argv[1], "compact") == 0) { ComPtr vdi; /* the hard disk image might not be registered */ if (!hardDisk) { virtualBox->OpenVirtualDiskImage(Bstr(argv[0]), vdi.asOutParam()); if (!hardDisk) { return errorArgument("Hard disk image not found"); } } else vdi = hardDisk; if (!vdi) return errorArgument("Invalid hard disk type. The command only works on VDI files\n"); Bstr fileName; vdi->COMGETTER(FilePath)(fileName.asOutParam()); /* close the file */ hardDisk = NULL; vdi = NULL; unsigned uProcent; RTPrintf("Shrinking '%lS': 0%%", fileName.raw()); int vrc = VDIShrinkImage(Utf8Str(fileName).raw(), vdiProgressCallback, &uProcent); if (VBOX_FAILURE(vrc)) { RTPrintf("Error while shrinking hard disk image: %Vrc\n", vrc); rc = E_FAIL; } } else { return errorSyntax(USAGE_MODIFYVDI, "Invalid parameter '%s'", Utf8Str(argv[1]).raw()); } return SUCCEEDED(rc) ? 0 : 1; } static int handleCloneVDI(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; /* source VDI and target path */ if (argc != 2) { return errorSyntax(USAGE_CLONEVDI, "Incorrect number of parameters"); } /* first guess is that it's a UUID */ Guid uuid(argv[0]); ComPtr hardDisk; rc = virtualBox->GetHardDisk(uuid, hardDisk.asOutParam()); if (!hardDisk) { /* not successful? Then it must be a filename */ ComPtr vdi; CHECK_ERROR(virtualBox, OpenVirtualDiskImage(Bstr(argv[0]), vdi.asOutParam())); hardDisk = vdi; } if (hardDisk) { ComPtr vdiOut; ComPtr progress; CHECK_ERROR(hardDisk, CloneToImage(Bstr(argv[1]), vdiOut.asOutParam(), progress.asOutParam())); if (SUCCEEDED(rc)) { showProgress(progress); progress->COMGETTER(ResultCode)(&rc); if (FAILED(rc)) { com::ProgressErrorInfo info(progress); if (info.isBasicAvailable()) { RTPrintf("Error: failed to clone disk image. Error message: %lS\n", info.getText().raw()); } else { RTPrintf("Error: failed to clone disk image. No error message available!\n"); } } } } return SUCCEEDED(rc) ? 0 : 1; } static int handleConvertDDImage(int argc, char *argv[]) { #ifdef __LINUX__ const bool fReadFromStdIn = !strcmp(argv[0], "stdin"); #else const bool fReadFromStdIn = false; #endif if ((!fReadFromStdIn && argc != 2) || (fReadFromStdIn && argc != 3)) return errorSyntax(USAGE_CONVERTDD, "Incorrect number of parameters"); RTPrintf("Converting VDI: from DD image file=\"%s\" to file=\"%s\"...\n", argv[0], argv[1]); /* open raw image file. */ RTFILE File; int rc = VINF_SUCCESS; if (fReadFromStdIn) File = 0; else rc = RTFileOpen(&File, argv[0], RTFILE_O_OPEN | RTFILE_O_READ | RTFILE_O_DENY_WRITE); if (VBOX_FAILURE(rc)) { RTPrintf("File=\"%s\" open error: %Rrf\n", argv[0], rc); return rc; } uint64_t cbFile; /* get image size. */ if (fReadFromStdIn) cbFile = RTStrToUInt64(argv[2]); else rc = RTFileGetSize(File, &cbFile); if (VBOX_SUCCESS(rc)) { RTPrintf("Creating fixed image with size %RU64Bytes (%RU64MB)...\n", cbFile, (cbFile + _1M - 1) / _1M); char pszComment[256]; RTStrPrintf(pszComment, sizeof(pszComment), "Converted image from %s", argv[0]); rc = VDICreateBaseImage(argv[1], VDI_IMAGE_TYPE_FIXED, cbFile, pszComment, NULL, NULL); if (VBOX_SUCCESS(rc)) { PVDIDISK pVdi = VDIDiskCreate(); rc = VDIDiskOpenImage(pVdi, argv[1], VDI_OPEN_FLAGS_NORMAL); if (VBOX_SUCCESS(rc)) { /* alloc work buffer. */ void *pvBuf = RTMemAlloc(VDIDiskGetBufferSize(pVdi)); if (pvBuf) { uint64_t off = 0; while (off < cbFile) { unsigned cbRead = 0; rc = RTFileRead(File, pvBuf, VDIDiskGetBufferSize(pVdi), &cbRead); if (VBOX_FAILURE(rc) || !cbRead) break; rc = VDIDiskWrite(pVdi, off, pvBuf, cbRead); if (VBOX_FAILURE(rc)) break; off += cbRead; } RTMemFree(pvBuf); } else rc = VERR_NO_MEMORY; VDIDiskCloseImage(pVdi); } if (VBOX_FAILURE(rc)) { /* delete image on error */ RTPrintf("Failed (%Vrc)!\n", rc); VDIDeleteImage(argv[1]); } } else RTPrintf("Failed to create output file (%Vrc)!\n", rc); } RTFileClose(File); return rc; } static int handleAddiSCSIDisk(int argc, char *argv[], ComPtr aVirtualBox, ComPtr aSession) { HRESULT rc; Bstr server; Bstr target; uint16_t port = UINT16_MAX; uint64_t lun = UINT64_MAX; Bstr username; Bstr password; Bstr comment; /* at least server and target */ if (argc < 4) { return errorSyntax(USAGE_ADDISCSIDISK, "Not enough parameters"); } /* let's have a closer look at the arguments */ for (int i = 0; i < argc; i++) { if (strcmp(argv[i], "-server") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; server = argv[i]; } else if (strcmp(argv[i], "-target") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; target = argv[i]; } else if (strcmp(argv[i], "-port") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; port = atoi(argv[i]); } else if (strcmp(argv[i], "-lun") == 0) { /** @todo move the LUN encoding algorithm into IISCSIHardDisk, add decoding */ if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; char *pszNext; int rc = RTStrToUInt64Ex(argv[i], &pszNext, 0, &lun); if (VBOX_FAILURE(rc) || *pszNext != '\0' || lun >= 16384) return errorArgument("Invalid LUN number '%s'", argv[i]); if (lun <= 255) { /* Assume bus identifier = 0. */ lun = (lun << 48); /* uses peripheral device addressing method */ } else { /* Check above already limited the LUN to 14 bits. */ lun = (lun << 48) | RT_BIT_64(62); /* uses flat space addressing method */ } } else if (strcmp(argv[i], "-encodedlun") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; char *pszNext; int rc = RTStrToUInt64Ex(argv[i], &pszNext, 0, &lun); if (VBOX_FAILURE(rc) || *pszNext != '\0') return errorArgument("Invalid encoded LUN number '%s'", argv[i]); } else if (strcmp(argv[i], "-username") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; username = argv[i]; } else if (strcmp(argv[i], "-password") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; password = argv[i]; } else if (strcmp(argv[i], "-comment") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; comment = argv[i]; } else { return errorSyntax(USAGE_ADDISCSIDISK, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); } } /* check for required options */ if (!server || !target) { return errorSyntax(USAGE_ADDISCSIDISK, "Parameters -server and -target are required"); } ComPtr hardDisk; CHECK_ERROR(aVirtualBox, CreateHardDisk(HardDiskStorageType_ISCSIHardDisk, hardDisk.asOutParam())); if (SUCCEEDED(rc) && hardDisk) { CHECK_ERROR(hardDisk, COMSETTER(Description)(comment)); ComPtr iSCSIDisk = hardDisk; CHECK_ERROR(iSCSIDisk, COMSETTER(Server)(server)); if (port != UINT16_MAX) CHECK_ERROR(iSCSIDisk, COMSETTER(Port)(port)); CHECK_ERROR(iSCSIDisk, COMSETTER(Target)(target)); if (lun != UINT64_MAX) CHECK_ERROR(iSCSIDisk, COMSETTER(Lun)(lun)); CHECK_ERROR(iSCSIDisk, COMSETTER(UserName)(username)); CHECK_ERROR(iSCSIDisk, COMSETTER(Password)(password)); if (SUCCEEDED(rc)) { CHECK_ERROR(aVirtualBox, RegisterHardDisk(hardDisk)); } if (SUCCEEDED(rc)) { Guid guid; CHECK_ERROR(hardDisk, COMGETTER(Id)(guid.asOutParam())); RTPrintf("iSCSI disk created. UUID: %s\n", guid.toString().raw()); } } return SUCCEEDED(rc) ? 0 : 1; } static int handleCreateVM(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; Bstr baseFolder; Bstr settingsFile; Bstr name; bool fRegister = false; for (int i = 0; i < argc; i++) { if (strcmp(argv[i], "-basefolder") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; baseFolder = argv[i]; } else if (strcmp(argv[i], "-settingsfile") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; settingsFile = argv[i]; } else if (strcmp(argv[i], "-name") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; name = argv[i]; } else if (strcmp(argv[i], "-register") == 0) { fRegister = true; } else { return errorSyntax(USAGE_CREATEVM, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); } } if (!name) { return errorSyntax(USAGE_CREATEVM, "Parameter -name is required"); } if (!!baseFolder && !!settingsFile) { return errorSyntax(USAGE_CREATEVM, "Either -basefolder or -settingsfile may be specified"); } do { ComPtr machine; if (!settingsFile) CHECK_ERROR_BREAK(virtualBox, CreateMachine(baseFolder, name, machine.asOutParam())); else CHECK_ERROR_BREAK(virtualBox, CreateLegacyMachine(settingsFile, name, machine.asOutParam())); CHECK_ERROR_BREAK(machine, SaveSettings()); if (fRegister) { CHECK_ERROR_BREAK(virtualBox, RegisterMachine(machine)); } Guid uuid; CHECK_ERROR_BREAK(machine, COMGETTER(Id)(uuid.asOutParam())); CHECK_ERROR_BREAK(machine, COMGETTER(SettingsFilePath)(settingsFile.asOutParam())); RTPrintf("Virtual machine '%ls' is created%s.\n" "UUID: %s\n" "Settings file: '%ls'\n", name.raw(), fRegister ? " and registered" : "", uuid.toString().raw(), settingsFile.raw()); } while (0); return SUCCEEDED(rc) ? 0 : 1; } /** * Parses a NIC number. * * @returns Valid nic number on success. * @returns 0 if invalid nic. All necesary bitching has been done. * @param psz Pointer to the nic number. */ static unsigned parseNicNum(const char *psz, unsigned cMaxNics) { uint32_t u32; char *pszNext; int rc = RTStrToUInt32Ex(psz, &pszNext, 10, &u32); if ( VBOX_SUCCESS(rc) && *pszNext == '\0' && u32 >= 1 && u32 <= cMaxNics) return (unsigned)u32; errorArgument("Invalid NIC number '%s'", psz); return 0; } static int handleModifyVM(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; Bstr name; Bstr ostype; ULONG memorySize = 0; ULONG vramSize = 0; char *acpi = NULL; char *hwvirtex = NULL; char *ioapic = NULL; char *bioslogofadein = NULL; char *bioslogofadeout = NULL; uint32_t bioslogodisplaytime = ~0; char *bioslogoimagepath = NULL; char *biosbootmenumode = NULL; DeviceType_T bootDevice[4]; int bootDeviceChanged[4] = { false }; char *hdds[4] = {0}; char *dvd = NULL; char *dvdpassthrough = NULL; char *floppy = NULL; char *audio = NULL; char *clipboard = NULL; #ifdef VBOX_VRDP char *vrdp = NULL; uint16_t vrdpport = UINT16_MAX; char *vrdpaddress = NULL; char *vrdpauthtype = NULL; #endif int fUsbEnabled = -1; char *snapshotFolder = NULL; /* VM ID + at least one parameter + value */ if (argc < 3) { return errorSyntax(USAGE_MODIFYVM, "Not enough parameters"); } /* Get the number of network adapters */ ULONG NetworkAdapterCount = 0; { ComPtr info; CHECK_ERROR_RET (virtualBox, COMGETTER(SystemProperties) (info.asOutParam()), 1); CHECK_ERROR_RET (info, COMGETTER(NetworkAdapterCount) (&NetworkAdapterCount), 1); } std::vector nics (NetworkAdapterCount, 0); std::vector nictype (NetworkAdapterCount, 0); std::vector cableconnected (NetworkAdapterCount, 0); std::vector nictrace (NetworkAdapterCount, 0); std::vector nictracefile (NetworkAdapterCount, 0); std::vector hostifdev (NetworkAdapterCount, 0); std::vector intnet (NetworkAdapterCount, 0); #ifdef __LINUX__ std::vector tapsetup (NetworkAdapterCount, 0); std::vector tapterm (NetworkAdapterCount, 0); #endif std::vector macs (NetworkAdapterCount, 0); for (int i = 1; i < argc; i++) { if (strcmp(argv[i], "-name") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; name = argv[i]; } else if (strcmp(argv[i], "-ostype") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; ostype = argv[i]; } else if (strcmp(argv[i], "-memory") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; memorySize = atoi(argv[i]); } else if (strcmp(argv[i], "-vram") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; vramSize = atoi(argv[i]); } else if (strcmp(argv[i], "-acpi") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; acpi = argv[i]; } else if (strcmp(argv[i], "-ioapic") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; ioapic = argv[i]; } else if (strcmp(argv[i], "-hwvirtex") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; hwvirtex = argv[i]; } else if (strcmp(argv[i], "-bioslogofadein") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; bioslogofadein = argv[i]; } else if (strcmp(argv[i], "-bioslogofadeout") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; bioslogofadeout = argv[i]; } else if (strcmp(argv[i], "-bioslogodisplaytime") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; bioslogodisplaytime = atoi(argv[i]); } else if (strcmp(argv[i], "-bioslogoimagepath") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; bioslogoimagepath = argv[i]; } else if (strcmp(argv[i], "-biosbootmenu") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; biosbootmenumode = argv[i]; } else if (strncmp(argv[i], "-boot", 5) == 0) { ULONG n = 0; if (!argv[i][5]) { return errorSyntax(USAGE_MODIFYVM, "Missing boot slot number in '%s'", argv[i]); } if ((n = strtoul(&argv[i][5], NULL, 10)) < 1) { return errorSyntax(USAGE_MODIFYVM, "Invalid boot slot number in '%s'", argv[i]); } if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; if (strcmp(argv[i], "none") == 0) { bootDevice[n - 1] = DeviceType_NoDevice; } else if (strcmp(argv[i], "floppy") == 0) { bootDevice[n - 1] = DeviceType_FloppyDevice; } else if (strcmp(argv[i], "dvd") == 0) { bootDevice[n - 1] = DeviceType_DVDDevice; } else if (strcmp(argv[i], "disk") == 0) { bootDevice[n - 1] = DeviceType_HardDiskDevice; } else if (strcmp(argv[i], "net") == 0) { bootDevice[n - 1] = DeviceType_NetworkDevice; } else { return errorArgument("Invalid boot device '%s'", argv[i]); } bootDeviceChanged[n - 1] = true; } else if (strcmp(argv[i], "-hda") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; hdds[0] = argv[i]; } else if (strcmp(argv[i], "-hdb") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; hdds[1] = argv[i]; } else if (strcmp(argv[i], "-hdd") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; hdds[2] = argv[i]; } else if (strcmp(argv[i], "-dvd") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; dvd = argv[i]; } else if (strcmp(argv[i], "-dvdpassthrough") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; dvdpassthrough = argv[i]; } else if (strcmp(argv[i], "-floppy") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; floppy = argv[i]; } else if (strcmp(argv[i], "-audio") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; audio = argv[i]; } else if (strcmp(argv[i], "-clipboard") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; clipboard = argv[i]; } else if (strncmp(argv[i], "-cableconnected", 15) == 0) { unsigned n = parseNicNum(&argv[i][15], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } cableconnected[n - 1] = argv[i + 1]; i++; } /* watch for the right order of these -nic* comparisons! */ else if (strncmp(argv[i], "-nictracefile", 13) == 0) { unsigned n = parseNicNum(&argv[i][13], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } nictracefile[n - 1] = argv[i + 1]; i++; } else if (strncmp(argv[i], "-nictrace", 9) == 0) { unsigned n = parseNicNum(&argv[i][9], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } nictrace[n - 1] = argv[i + 1]; i++; } else if (strncmp(argv[i], "-nictype", 8) == 0) { unsigned n = parseNicNum(&argv[i][8], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } nictype[n - 1] = argv[i + 1]; i++; } else if (strncmp(argv[i], "-nic", 4) == 0) { unsigned n = parseNicNum(&argv[i][4], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } nics[n - 1] = argv[i + 1]; i++; } else if (strncmp(argv[i], "-hostifdev", 10) == 0) { unsigned n = parseNicNum(&argv[i][10], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } hostifdev[n - 1] = argv[i + 1]; i++; } else if (strncmp(argv[i], "-intnet", 7) == 0) { unsigned n = parseNicNum(&argv[i][7], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } intnet[n - 1] = argv[i + 1]; i++; } #ifdef __LINUX__ else if (strncmp(argv[i], "-tapsetup", 9) == 0) { unsigned n = parseNicNum(&argv[i][9], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } tapsetup[n - 1] = argv[i + 1]; i++; } else if (strncmp(argv[i], "-tapterminate", 13) == 0) { unsigned n = parseNicNum(&argv[i][13], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } tapterm[n - 1] = argv[i + 1]; i++; } #endif /* __LINUX__ */ else if (strncmp(argv[i], "-macaddress", 11) == 0) { unsigned n = parseNicNum(&argv[i][11], NetworkAdapterCount); if (!n) return 1; if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } macs[n - 1] = argv[i + 1]; i++; } #ifdef VBOX_VRDP else if (strcmp(argv[i], "-vrdp") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; vrdp = argv[i]; } else if (strcmp(argv[i], "-vrdpport") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; if (strcmp(argv[i], "default") == 0) vrdpport = 0; else vrdpport = atoi(argv[i]); } else if (strcmp(argv[i], "-vrdpaddress") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; vrdpaddress = argv[i]; } else if (strcmp(argv[i], "-vrdpauthtype") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; vrdpauthtype = argv[i]; } #endif /* VBOX_VRDP */ else if (strcmp(argv[i], "-usb") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; if (strcmp(argv[i], "on") == 0 || strcmp(argv[i], "enable") == 0) fUsbEnabled = 1; else if (strcmp(argv[i], "off") == 0 || strcmp(argv[i], "disable") == 0) fUsbEnabled = 0; else return errorArgument("Invalid -usb argument '%s'", argv[i]); } else if (strcmp(argv[i], "-snapshotfolder") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; snapshotFolder = argv[i]; } else { return errorSyntax(USAGE_MODIFYVM, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); } } /* try to find the given machine */ ComPtr machine; Guid uuid (argv[0]); if (!uuid.isEmpty()) { CHECK_ERROR (virtualBox, GetMachine (uuid, machine.asOutParam())); } else { CHECK_ERROR (virtualBox, FindMachine(Bstr(argv[0]), machine.asOutParam())); if (SUCCEEDED (rc)) machine->COMGETTER(Id)(uuid.asOutParam()); } if (FAILED (rc)) return 1; /* open a session for the VM */ CHECK_ERROR_RET (virtualBox, OpenSession(session, uuid), 1); do { /* get the mutable session machine */ session->COMGETTER(Machine)(machine.asOutParam()); ComPtr biosSettings; machine->COMGETTER(BIOSSettings)(biosSettings.asOutParam()); if (name) CHECK_ERROR(machine, COMSETTER(Name)(name)); if (ostype) { ComPtr guestOSType; CHECK_ERROR(virtualBox, FindGuestOSType(ostype, guestOSType.asOutParam())); if (SUCCEEDED(rc) && guestOSType) { CHECK_ERROR(machine, COMSETTER(OSType)(guestOSType)); } else { errorArgument("Invalid guest OS type '%s'", Utf8Str(ostype).raw()); rc = E_FAIL; break; } } if (memorySize > 0) CHECK_ERROR(machine, COMSETTER(MemorySize)(memorySize)); if (vramSize > 0) CHECK_ERROR(machine, COMSETTER(VRAMSize)(vramSize)); if (acpi) { if (strcmp(acpi, "on") == 0) { CHECK_ERROR(biosSettings, COMSETTER(ACPIEnabled)(true)); } else if (strcmp(acpi, "off") == 0) { CHECK_ERROR(biosSettings, COMSETTER(ACPIEnabled)(false)); } else { errorArgument("Invalid -acpi argument '%s'", acpi); rc = E_FAIL; break; } } if (ioapic) { if (strcmp(ioapic, "on") == 0) { CHECK_ERROR(biosSettings, COMSETTER(IOAPICEnabled)(true)); } else if (strcmp(ioapic, "off") == 0) { CHECK_ERROR(biosSettings, COMSETTER(IOAPICEnabled)(false)); } else { errorArgument("Invalid -ioapic argument '%s'", ioapic); rc = E_FAIL; break; } } if (hwvirtex) { if (strcmp(hwvirtex, "on") == 0) { CHECK_ERROR(machine, COMSETTER(HWVirtExEnabled)(TriStateBool_True)); } else if (strcmp(hwvirtex, "off") == 0) { CHECK_ERROR(machine, COMSETTER(HWVirtExEnabled)(TriStateBool_False)); } else if (strcmp(hwvirtex, "default") == 0) { CHECK_ERROR(machine, COMSETTER(HWVirtExEnabled)(TriStateBool_Default)); } else { errorArgument("Invalid -hwvirtex argument '%s'", hwvirtex); rc = E_FAIL; break; } } if (bioslogofadein) { if (strcmp(bioslogofadein, "on") == 0) { CHECK_ERROR(biosSettings, COMSETTER(LogoFadeIn)(true)); } else if (strcmp(bioslogofadein, "off") == 0) { CHECK_ERROR(biosSettings, COMSETTER(LogoFadeIn)(false)); } else { errorArgument("Invalid -bioslogofadein argument '%s'", bioslogofadein); rc = E_FAIL; break; } } if (bioslogofadeout) { if (strcmp(bioslogofadeout, "on") == 0) { CHECK_ERROR(biosSettings, COMSETTER(LogoFadeOut)(true)); } else if (strcmp(bioslogofadeout, "off") == 0) { CHECK_ERROR(biosSettings, COMSETTER(LogoFadeOut)(false)); } else { errorArgument("Invalid -bioslogofadeout argument '%s'", bioslogofadeout); rc = E_FAIL; break; } } if (bioslogodisplaytime != ~0U) { CHECK_ERROR(biosSettings, COMSETTER(LogoDisplayTime)(bioslogodisplaytime)); } if (bioslogoimagepath) { CHECK_ERROR(biosSettings, COMSETTER(LogoImagePath)(Bstr(bioslogoimagepath))); } if (biosbootmenumode) { if (strcmp(biosbootmenumode, "disabled") == 0) CHECK_ERROR(biosSettings, COMSETTER(BootMenuMode)(BIOSBootMenuMode_Disabled)); else if (strcmp(biosbootmenumode, "menuonly") == 0) CHECK_ERROR(biosSettings, COMSETTER(BootMenuMode)(BIOSBootMenuMode_MenuOnly)); else if (strcmp(biosbootmenumode, "messageandmenu") == 0) CHECK_ERROR(biosSettings, COMSETTER(BootMenuMode)(BIOSBootMenuMode_MessageAndMenu)); else { errorArgument("Invalid -biosbootmenu argument '%s'", biosbootmenumode); rc = E_FAIL; break; } } for (int curBootDev = 0; curBootDev < 4; curBootDev++) { if (bootDeviceChanged[curBootDev]) CHECK_ERROR(machine, SetBootOrder (curBootDev + 1, bootDevice[curBootDev])); } if (hdds[0]) { if (strcmp(hdds[0], "none") == 0) { machine->DetachHardDisk(DiskControllerType_IDE0Controller, 0); } else { /* first guess is that it's a UUID */ Guid uuid(hdds[0]); ComPtr hardDisk; rc = virtualBox->GetHardDisk(uuid, hardDisk.asOutParam()); /* not successful? Then it must be a filename */ if (!hardDisk) { ComPtr vdi; CHECK_ERROR(virtualBox, OpenVirtualDiskImage(Bstr(hdds[0]), vdi.asOutParam())); if (SUCCEEDED(rc) && vdi) { hardDisk = vdi; /* first check if it's already registered */ Guid hddUUID; hardDisk->COMGETTER(Id)(hddUUID.asOutParam()); ComPtr registeredHDD; rc = virtualBox->GetHardDisk(hddUUID, registeredHDD.asOutParam()); if (SUCCEEDED(rc) && registeredHDD) hardDisk = registeredHDD; else { /* it has to be registered */ CHECK_ERROR(virtualBox, RegisterHardDisk(hardDisk)); if (FAILED(rc)) break; } } } if (hardDisk) { hardDisk->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(machine, AttachHardDisk(uuid, DiskControllerType_IDE0Controller, 0)); } else rc = E_FAIL; if (FAILED(rc)) break; } } if (hdds[1]) { if (strcmp(hdds[1], "none") == 0) { machine->DetachHardDisk(DiskControllerType_IDE0Controller, 1); } else { /* first guess is that it's a UUID */ Guid uuid(hdds[1]); ComPtr hardDisk; rc = virtualBox->GetHardDisk(uuid, hardDisk.asOutParam()); /* not successful? Then it must be a filename */ if (!hardDisk) { ComPtr vdi; CHECK_ERROR(virtualBox, OpenVirtualDiskImage(Bstr(hdds[1]), vdi.asOutParam())); if (SUCCEEDED(rc) && vdi) { hardDisk = vdi; /* first check if it's already registered */ Guid hddUUID; hardDisk->COMGETTER(Id)(hddUUID.asOutParam()); ComPtr registeredHDD; rc = virtualBox->GetHardDisk(hddUUID, registeredHDD.asOutParam()); if (SUCCEEDED(rc) && registeredHDD) hardDisk = registeredHDD; else { /* it has to be registered */ CHECK_ERROR(virtualBox, RegisterHardDisk(hardDisk)); if (FAILED(rc)) break; } } } if (hardDisk) { hardDisk->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(machine, AttachHardDisk(uuid, DiskControllerType_IDE0Controller, 1)); } else rc = E_FAIL; if (FAILED(rc)) break; } } if (hdds[2]) { if (strcmp(hdds[2], "none") == 0) { machine->DetachHardDisk(DiskControllerType_IDE1Controller, 1); } else { /* first guess is that it's a UUID */ Guid uuid(hdds[2]); ComPtr hardDisk; rc = virtualBox->GetHardDisk(uuid, hardDisk.asOutParam()); /* not successful? Then it must be a filename */ if (!hardDisk) { ComPtr vdi; CHECK_ERROR(virtualBox, OpenVirtualDiskImage(Bstr(hdds[2]), vdi.asOutParam())); if (SUCCEEDED(rc) && vdi) { hardDisk = vdi; /* first check if it's already registered */ Guid hddUUID; hardDisk->COMGETTER(Id)(hddUUID.asOutParam()); ComPtr registeredHDD; rc = virtualBox->GetHardDisk(hddUUID, registeredHDD.asOutParam()); if (SUCCEEDED(rc) && registeredHDD) hardDisk = registeredHDD; else { /* it has to be registered */ CHECK_ERROR(virtualBox, RegisterHardDisk(hardDisk)); if (FAILED(rc)) break; } } } if (hardDisk) { hardDisk->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(machine, AttachHardDisk(uuid, DiskControllerType_IDE1Controller, 1)); } else rc = E_FAIL; if (FAILED(rc)) break; } } if (dvd) { ComPtr dvdDrive; machine->COMGETTER(DVDDrive)(dvdDrive.asOutParam()); ASSERT(dvdDrive); /* unmount? */ if (strcmp(dvd, "none") == 0) { CHECK_ERROR(dvdDrive, Unmount()); } /* host drive? */ else if (strncmp(dvd, "host:", 5) == 0) { ComPtr host; CHECK_ERROR(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr hostDVDs; CHECK_ERROR(host, COMGETTER(DVDDrives)(hostDVDs.asOutParam())); ComPtr hostDVDDrive; rc = hostDVDs->FindByName(Bstr(dvd + 5), hostDVDDrive.asOutParam()); if (!hostDVDDrive) { errorArgument("Invalid host DVD drive name"); rc = E_FAIL; break; } CHECK_ERROR(dvdDrive, CaptureHostDrive(hostDVDDrive)); } else { /* first assume it's a UUID */ Guid uuid(dvd); ComPtr dvdImage; rc = virtualBox->GetDVDImage(uuid, dvdImage.asOutParam()); if (FAILED(rc) || !dvdImage) { /* must be a filename, check if it's in the collection */ ComPtr dvdImages; virtualBox->COMGETTER(DVDImages)(dvdImages.asOutParam()); rc = dvdImages->FindByPath(Bstr(dvd), dvdImage.asOutParam()); /* not registered, do that on the fly */ if (!dvdImage) { Guid emptyUUID; CHECK_ERROR(virtualBox, OpenDVDImage(Bstr(dvd), emptyUUID, dvdImage.asOutParam())); if (SUCCEEDED(rc) && dvdImage) { /* time to register the image */ CHECK_ERROR(virtualBox, RegisterDVDImage(dvdImage)); } } } if (!dvdImage) { rc = E_FAIL; break; } dvdImage->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(dvdDrive, MountImage(uuid)); } } if (dvdpassthrough) { ComPtr dvdDrive; machine->COMGETTER(DVDDrive)(dvdDrive.asOutParam()); ASSERT(dvdDrive); CHECK_ERROR(dvdDrive, COMSETTER(Passthrough)(strcmp(dvdpassthrough, "on") == 0)); } if (floppy) { ComPtr floppyDrive; machine->COMGETTER(FloppyDrive)(floppyDrive.asOutParam()); ASSERT(floppyDrive); /* disable? */ if (strcmp(floppy, "disabled") == 0) { /* disable the controller */ CHECK_ERROR(floppyDrive, COMSETTER(Enabled)(false)); } else { /* enable the controller */ CHECK_ERROR(floppyDrive, COMSETTER(Enabled)(true)); /* unmount? */ if (strcmp(floppy, "empty") == 0) { CHECK_ERROR(floppyDrive, Unmount()); } /* host drive? */ else if (strncmp(floppy, "host:", 5) == 0) { ComPtr host; CHECK_ERROR(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr hostFloppies; CHECK_ERROR(host, COMGETTER(FloppyDrives)(hostFloppies.asOutParam())); ComPtr hostFloppyDrive; rc = hostFloppies->FindByName(Bstr(floppy + 5), hostFloppyDrive.asOutParam()); if (!hostFloppyDrive) { errorArgument("Invalid host floppy drive name"); rc = E_FAIL; break; } CHECK_ERROR(floppyDrive, CaptureHostDrive(hostFloppyDrive)); } else { /* first assume it's a UUID */ Guid uuid(floppy); ComPtr floppyImage; rc = virtualBox->GetFloppyImage(uuid, floppyImage.asOutParam()); if (FAILED(rc) || !floppyImage) { /* must be a filename */ Guid emptyUUID; CHECK_ERROR(virtualBox, OpenFloppyImage(Bstr(floppy), emptyUUID, floppyImage.asOutParam())); if (SUCCEEDED(rc) && floppyImage) { /** @todo first iterate through the collection and try to find the image */ /* time to register the image */ CHECK_ERROR(virtualBox, RegisterFloppyImage(floppyImage)); } } if (!floppyImage) { rc = E_FAIL; break; } floppyImage->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(floppyDrive, MountImage(uuid)); } } } if (audio) { ComPtr audioAdapter; machine->COMGETTER(AudioAdapter)(audioAdapter.asOutParam()); ASSERT(audioAdapter); /* disable? */ if (strcmp(audio, "none") == 0) { CHECK_ERROR(audioAdapter, COMSETTER(Enabled)(false)); } else if (strcmp(audio, "null") == 0) { CHECK_ERROR(audioAdapter, COMSETTER(AudioDriver)(AudioDriverType_NullAudioDriver)); CHECK_ERROR(audioAdapter, COMSETTER(Enabled)(true)); } #ifdef __WIN__ else if (strcmp(audio, "winmm") == 0) { CHECK_ERROR(audioAdapter, COMSETTER(AudioDriver)(AudioDriverType_WINMMAudioDriver)); CHECK_ERROR(audioAdapter, COMSETTER(Enabled)(true)); } else if (strcmp(audio, "dsound") == 0) { CHECK_ERROR(audioAdapter, COMSETTER(AudioDriver)(AudioDriverType_DSOUNDAudioDriver)); CHECK_ERROR(audioAdapter, COMSETTER(Enabled)(true)); } #endif /* __WIN__ */ #ifdef __LINUX__ else if (strcmp(audio, "oss") == 0) { CHECK_ERROR(audioAdapter, COMSETTER(AudioDriver)(AudioDriverType_OSSAudioDriver)); CHECK_ERROR(audioAdapter, COMSETTER(Enabled)(true)); } # ifdef VBOX_WITH_ALSA else if (strcmp(audio, "alsa") == 0) { CHECK_ERROR(audioAdapter, COMSETTER(AudioDriver)(AudioDriverType_ALSAAudioDriver)); CHECK_ERROR(audioAdapter, COMSETTER(Enabled)(true)); } # endif #endif /* !__LINUX__ */ #ifdef __DARWIN__ else if (strcmp(audio, "coreaudio") == 0) { CHECK_ERROR(audioAdapter, COMSETTER(AudioDriver)(AudioDriverType_CoreAudioDriver)); CHECK_ERROR(audioAdapter, COMSETTER(Enabled)(true)); } #endif /* !__DARWIN__ */ else { errorArgument("Invalid -audio argument '%s'", audio); rc = E_FAIL; break; } } /* Shared clipboard state */ if (clipboard) { /* ComPtr clipboardMode; machine->COMGETTER(ClipboardMode)(clipboardMode.asOutParam()); ASSERT(clipboardMode); */ if (strcmp(clipboard, "disabled") == 0) { CHECK_ERROR(machine, COMSETTER(ClipboardMode)(ClipboardMode_ClipDisabled)); } else if (strcmp(clipboard, "hosttoguest") == 0) { CHECK_ERROR(machine, COMSETTER(ClipboardMode)(ClipboardMode_ClipHostToGuest)); } else if (strcmp(clipboard, "guesttohost") == 0) { CHECK_ERROR(machine, COMSETTER(ClipboardMode)(ClipboardMode_ClipGuestToHost)); } else if (strcmp(clipboard, "bidirectional") == 0) { CHECK_ERROR(machine, COMSETTER(ClipboardMode)(ClipboardMode_ClipBidirectional)); } else { errorArgument("Invalid -clipboard argument '%s'", clipboard); rc = E_FAIL; break; } } /* iterate through all possible NICs */ for (ULONG n = 0; n < NetworkAdapterCount; n ++) { ComPtr nic; CHECK_ERROR_RET (machine, GetNetworkAdapter (n, nic.asOutParam()), 1); ASSERT(nic); /* something about the NIC? */ if (nics[n]) { if (strcmp(nics[n], "none") == 0) { CHECK_ERROR_RET(nic, COMSETTER(Enabled) (FALSE), 1); } else if (strcmp(nics[n], "null") == 0) { CHECK_ERROR_RET(nic, COMSETTER(Enabled) (TRUE), 1); CHECK_ERROR_RET(nic, Detach(), 1); } else if (strcmp(nics[n], "nat") == 0) { CHECK_ERROR_RET(nic, COMSETTER(Enabled) (TRUE), 1); CHECK_ERROR_RET(nic, AttachToNAT(), 1); } else if (strcmp(nics[n], "hostif") == 0) { CHECK_ERROR_RET(nic, COMSETTER(Enabled) (TRUE), 1); CHECK_ERROR_RET(nic, AttachToHostInterface(), 1); } else if (strcmp(nics[n], "intnet") == 0) { CHECK_ERROR_RET(nic, COMSETTER(Enabled) (TRUE), 1); CHECK_ERROR_RET(nic, AttachToInternalNetwork(), 1); } else { errorArgument("Invalid type '%s' specfied for NIC %lu", nics[n], n + 1); rc = E_FAIL; break; } } /* something about the NIC type? */ if (nictype[n]) { if (strcmp(nictype[n], "Am79C970A") == 0) { CHECK_ERROR_RET(nic, COMSETTER(AdapterType)(NetworkAdapterType_NetworkAdapterAm79C970A), 1); } else if (strcmp(nictype[n], "Am79C973") == 0) { CHECK_ERROR_RET(nic, COMSETTER(AdapterType)(NetworkAdapterType_NetworkAdapterAm79C973), 1); } else { errorArgument("Invalid NIC type '%s' specified for NIC %lu", nictype[n], n + 1); rc = E_FAIL; break; } } /* something about the MAC address? */ if (macs[n]) { /* generate one? */ if (strcmp(macs[n], "auto") == 0) { CHECK_ERROR_RET(nic, COMSETTER(MACAddress)(NULL), 1); } else { CHECK_ERROR_RET(nic, COMSETTER(MACAddress)(Bstr(macs[n])), 1); } } /* the link status flag? */ if (cableconnected[n]) { if (strcmp(cableconnected[n], "on") == 0) { CHECK_ERROR_RET(nic, COMSETTER(CableConnected)(TRUE), 1); } else if (strcmp(cableconnected[n], "off") == 0) { CHECK_ERROR_RET(nic, COMSETTER(CableConnected)(FALSE), 1); } else { errorArgument("Invalid -cableconnected%lu argument '%s'", n + 1, cableconnected[n]); rc = E_FAIL; break; } } /* the trace flag? */ if (nictrace[n]) { if (strcmp(nictrace[n], "on") == 0) { CHECK_ERROR_RET(nic, COMSETTER(TraceEnabled)(TRUE), 1); } else if (strcmp(nictrace[n], "off") == 0) { CHECK_ERROR_RET(nic, COMSETTER(TraceEnabled)(FALSE), 1); } else { errorArgument("Invalid -nictrace%lu argument '%s'", n + 1, nictrace[n]); rc = E_FAIL; break; } } /* the tracefile flag? */ if (nictracefile[n]) { CHECK_ERROR_RET(nic, COMSETTER(TraceFile)(Bstr(nictracefile[n])), 1); } /* the host interface device? */ if (hostifdev[n]) { /* remove it? */ if (strcmp(hostifdev[n], "none") == 0) { CHECK_ERROR_RET(nic, COMSETTER(HostInterface)(NULL), 1); } else { CHECK_ERROR_RET(nic, COMSETTER(HostInterface)(Bstr(hostifdev[n])), 1); } } /* the internal network name? */ if (intnet[n]) { /* remove it? */ if (strcmp(intnet[n], "none") == 0) { CHECK_ERROR_RET(nic, COMSETTER(InternalNetwork)(NULL), 1); } else { CHECK_ERROR_RET(nic, COMSETTER(InternalNetwork)(Bstr(intnet[n])), 1); } } #ifdef __LINUX__ /* the TAP setup application? */ if (tapsetup[n]) { /* remove it? */ if (strcmp(tapsetup[n], "none") == 0) { CHECK_ERROR_RET(nic, COMSETTER(TAPSetupApplication)(NULL), 1); } else { CHECK_ERROR_RET(nic, COMSETTER(TAPSetupApplication)(Bstr(tapsetup[n])), 1); } } /* the TAP terminate application? */ if (tapterm[n]) { /* remove it? */ if (strcmp(tapterm[n], "none") == 0) { CHECK_ERROR_RET(nic, COMSETTER(TAPTerminateApplication)(NULL), 1); } else { CHECK_ERROR_RET(nic, COMSETTER(TAPTerminateApplication)(Bstr(tapterm[n])), 1); } } #endif /* __LINUX__ */ } if (FAILED(rc)) break; #ifdef VBOX_VRDP if (vrdp || (vrdpport != UINT16_MAX) || vrdpaddress || vrdpauthtype) { ComPtr vrdpServer; machine->COMGETTER(VRDPServer)(vrdpServer.asOutParam()); ASSERT(vrdpServer); if (vrdpServer) { if (vrdp) { if (strcmp(vrdp, "on") == 0) { CHECK_ERROR(vrdpServer, COMSETTER(Enabled)(true)); } else if (strcmp(vrdp, "off") == 0) { CHECK_ERROR(vrdpServer, COMSETTER(Enabled)(false)); } else { errorArgument("Invalid -vrdp argument '%s'", vrdp); rc = E_FAIL; break; } } if (vrdpport != UINT16_MAX) { CHECK_ERROR(vrdpServer, COMSETTER(Port)(vrdpport)); } if (vrdpaddress) { CHECK_ERROR(vrdpServer, COMSETTER(NetAddress)(Bstr(vrdpaddress))); } if (vrdpauthtype) { if (strcmp(vrdpauthtype, "null") == 0) { CHECK_ERROR(vrdpServer, COMSETTER(AuthType)(VRDPAuthType_VRDPAuthNull)); } else if (strcmp(vrdpauthtype, "external") == 0) { CHECK_ERROR(vrdpServer, COMSETTER(AuthType)(VRDPAuthType_VRDPAuthExternal)); } else if (strcmp(vrdpauthtype, "guest") == 0) { CHECK_ERROR(vrdpServer, COMSETTER(AuthType)(VRDPAuthType_VRDPAuthGuest)); } else { errorArgument("Invalid -vrdpauthtype argument '%s'", vrdpauthtype); rc = E_FAIL; break; } } } } #endif /* VBOX_VRDP */ /* * USB enable/disable */ if (fUsbEnabled != -1) { ComPtr UsbCtl; CHECK_ERROR(machine, COMGETTER(USBController)(UsbCtl.asOutParam())); if (SUCCEEDED(rc)) { CHECK_ERROR(UsbCtl, COMSETTER(Enabled)(!!fUsbEnabled)); } } if (snapshotFolder) { if (strcmp(snapshotFolder, "default") == 0) { CHECK_ERROR(machine, COMSETTER(SnapshotFolder)(NULL)); } else { CHECK_ERROR(machine, COMSETTER(SnapshotFolder)(Bstr(snapshotFolder))); } } /* commit changes */ CHECK_ERROR(machine, SaveSettings()); } while (0); /* it's important to always close sessions */ session->Close(); return SUCCEEDED(rc) ? 0 : 1; } static int handleStartVM(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; if (argc < 1) { return errorSyntax(USAGE_STARTVM, "Not enough parameters"); } ComPtr machine; /* assume it's a UUID */ rc = virtualBox->GetMachine(Guid(argv[0]), machine.asOutParam()); if (FAILED(rc) || !machine) { /* must be a name */ CHECK_ERROR(virtualBox, FindMachine(Bstr(argv[0]), machine.asOutParam())); } if (machine) { Guid uuid; machine->COMGETTER(Id)(uuid.asOutParam()); /* default to GUI session type */ Bstr sessionType = "gui"; /* has a session type been specified? */ if ((argc > 2) && (strcmp(argv[1], "-type") == 0)) { if (strcmp(argv[2], "gui") == 0) { sessionType = "gui"; } else if (strcmp(argv[2], "vrdp") == 0) { sessionType = "vrdp"; } else if (strcmp(argv[2], "capture") == 0) { sessionType = "capture"; } else { return errorArgument("Invalid session type argument '%s'", argv[2]); } } ComPtr progress; CHECK_ERROR_RET(virtualBox, OpenRemoteSession(session, uuid, sessionType, progress.asOutParam()), rc); RTPrintf("Waiting for the remote session to open...\n"); CHECK_ERROR_RET(progress, WaitForCompletion (-1), 1); BOOL completed; CHECK_ERROR_RET(progress, COMGETTER(Completed)(&completed), rc); ASSERT(completed); HRESULT resultCode; CHECK_ERROR_RET(progress, COMGETTER(ResultCode)(&resultCode), rc); if (FAILED(resultCode)) { ComPtr errorInfo; CHECK_ERROR_RET(progress, COMGETTER(ErrorInfo)(errorInfo.asOutParam()), 1); ErrorInfo info (errorInfo); PRINT_ERROR_INFO(info); } else { RTPrintf("Remote session has been successfully opened.\n"); } } /* it's important to always close sessions */ session->Close(); return SUCCEEDED(rc) ? 0 : 1; } static int handleControlVM(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; if (argc < 2) { return errorSyntax(USAGE_CONTROLVM, "Not enough parameters"); } /* try to find the given machine */ ComPtr machine; Guid uuid (argv[0]); if (!uuid.isEmpty()) { CHECK_ERROR (virtualBox, GetMachine (uuid, machine.asOutParam())); } else { CHECK_ERROR (virtualBox, FindMachine (Bstr(argv[0]), machine.asOutParam())); if (SUCCEEDED (rc)) machine->COMGETTER(Id) (uuid.asOutParam()); } if (FAILED (rc)) return 1; /* open a session for the VM */ CHECK_ERROR_RET (virtualBox, OpenExistingSession (session, uuid), 1); do { /* get the associated console */ ComPtr console; CHECK_ERROR_BREAK (session, COMGETTER(Console)(console.asOutParam())); /* ... and session machine */ ComPtr sessionMachine; CHECK_ERROR_BREAK (session, COMGETTER(Machine)(sessionMachine.asOutParam())); /* which command? */ if (strcmp(argv[1], "pause") == 0) { CHECK_ERROR_BREAK (console, Pause()); } else if (strcmp(argv[1], "resume") == 0) { CHECK_ERROR_BREAK (console, Resume()); } else if (strcmp(argv[1], "reset") == 0) { CHECK_ERROR_BREAK (console, Reset()); } else if (strcmp(argv[1], "poweroff") == 0) { CHECK_ERROR_BREAK (console, PowerDown()); } else if (strcmp(argv[1], "savestate") == 0) { ComPtr progress; CHECK_ERROR_BREAK (console, SaveState(progress.asOutParam())); showProgress(progress); progress->COMGETTER(ResultCode)(&rc); if (FAILED(rc)) { com::ProgressErrorInfo info(progress); if (info.isBasicAvailable()) { RTPrintf("Error: failed to save machine state. Error message: %lS\n", info.getText().raw()); } else { RTPrintf("Error: failed to save machine state. No error message available!\n"); } } } else if (strcmp(argv[1], "acpipowerbutton") == 0) { CHECK_ERROR_BREAK (console, PowerButton()); } else if (strncmp(argv[1], "setlinkstate", 12) == 0) { /* Get the number of network adapters */ ULONG NetworkAdapterCount = 0; ComPtr info; CHECK_ERROR_BREAK (virtualBox, COMGETTER(SystemProperties) (info.asOutParam())); CHECK_ERROR_BREAK (info, COMGETTER(NetworkAdapterCount) (&NetworkAdapterCount)); unsigned n = parseNicNum(&argv[1][12], NetworkAdapterCount); if (!n) { rc = E_FAIL; break; } if (argc <= 1 + 1) { errorArgument("Missing argument to '%s'", argv[1]); rc = E_FAIL; break; } /* get the corresponding network adapter */ ComPtr adapter; CHECK_ERROR_BREAK (sessionMachine, GetNetworkAdapter(n - 1, adapter.asOutParam())); if (adapter) { if (strcmp(argv[2], "on") == 0) { CHECK_ERROR_BREAK (adapter, COMSETTER(CableConnected)(TRUE)); } else if (strcmp(argv[2], "off") == 0) { CHECK_ERROR_BREAK (adapter, COMSETTER(CableConnected)(FALSE)); } else { errorArgument("Invalid link state '%s'", Utf8Str(argv[2]).raw()); rc = E_FAIL; break; } } } else if (strcmp (argv[1], "usbattach") == 0 || strcmp (argv[1], "usbdetach") == 0) { if (argc < 3) { errorSyntax(USAGE_CONTROLVM, "Not enough parameters"); rc = E_FAIL; break; } bool attach = strcmp (argv[1], "usbattach") == 0; Guid usbId = argv [2]; if (usbId.isEmpty()) { // assume address if (attach) { ComPtr host; CHECK_ERROR_BREAK (virtualBox, COMGETTER(Host) (host.asOutParam())); ComPtr coll; CHECK_ERROR_BREAK (host, COMGETTER(USBDevices) (coll.asOutParam())); ComPtr dev; CHECK_ERROR_BREAK (coll, FindByAddress (Bstr (argv [2]), dev.asOutParam())); CHECK_ERROR_BREAK (dev, COMGETTER(Id) (usbId.asOutParam())); } else { ComPtr coll; CHECK_ERROR_BREAK (console, COMGETTER(USBDevices)(coll.asOutParam())); ComPtr dev; CHECK_ERROR_BREAK (coll, FindByAddress (Bstr (argv [2]), dev.asOutParam())); CHECK_ERROR_BREAK (dev, COMGETTER(Id) (usbId.asOutParam())); } } if (attach) CHECK_ERROR_BREAK (console, AttachUSBDevice (usbId)); else { ComPtr dev; CHECK_ERROR_BREAK (console, DetachUSBDevice (usbId, dev.asOutParam())); } } else if (strcmp(argv[1], "setvideomodehint") == 0) { if (argc != 5) { errorSyntax(USAGE_CONTROLVM, "Incorrect number of parameters"); rc = E_FAIL; break; } uint32_t xres = atoi(argv[2]); uint32_t yres = atoi(argv[3]); uint32_t bpp = atoi(argv[4]); ComPtr display; CHECK_ERROR_BREAK(console, COMGETTER(Display)(display.asOutParam())); CHECK_ERROR_BREAK(display, SetVideoModeHint(xres, yres, bpp)); } else if (strcmp(argv[1], "setcredentials") == 0) { bool fAllowLocalLogon = true; if (argc == 7) { if (strcmp(argv[5], "-allowlocallogon") != 0) { errorArgument("Invalid parameter '%s'", argv[5]); rc = E_FAIL; break; } if (strcmp(argv[6], "no") == 0) fAllowLocalLogon = false; } else if (argc != 5) { errorSyntax(USAGE_CONTROLVM, "Incorrect number of parameters"); rc = E_FAIL; break; } ComPtr guest; CHECK_ERROR_BREAK(console, COMGETTER(Guest)(guest.asOutParam())); CHECK_ERROR_BREAK(guest, SetCredentials(Bstr(argv[2]), Bstr(argv[3]), Bstr(argv[4]), fAllowLocalLogon)); } else if (strcmp(argv[1], "dvdattach") == 0) { if (argc != 3) { errorSyntax(USAGE_CONTROLVM, "Incorrect number of parameters"); rc = E_FAIL; break; } ComPtr dvdDrive; sessionMachine->COMGETTER(DVDDrive)(dvdDrive.asOutParam()); ASSERT(dvdDrive); /* unmount? */ if (strcmp(argv[2], "none") == 0) { CHECK_ERROR(dvdDrive, Unmount()); } /* host drive? */ else if (strncmp(argv[2], "host:", 5) == 0) { ComPtr host; CHECK_ERROR(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr hostDVDs; CHECK_ERROR(host, COMGETTER(DVDDrives)(hostDVDs.asOutParam())); ComPtr hostDVDDrive; rc = hostDVDs->FindByName(Bstr(argv[2] + 5), hostDVDDrive.asOutParam()); if (!hostDVDDrive) { errorArgument("Invalid host DVD drive name"); rc = E_FAIL; break; } CHECK_ERROR(dvdDrive, CaptureHostDrive(hostDVDDrive)); } else { /* first assume it's a UUID */ Guid uuid(argv[2]); ComPtr dvdImage; rc = virtualBox->GetDVDImage(uuid, dvdImage.asOutParam()); if (FAILED(rc) || !dvdImage) { /* must be a filename, check if it's in the collection */ ComPtr dvdImages; virtualBox->COMGETTER(DVDImages)(dvdImages.asOutParam()); rc = dvdImages->FindByPath(Bstr(argv[2]), dvdImage.asOutParam()); /* not registered, do that on the fly */ if (!dvdImage) { Guid emptyUUID; CHECK_ERROR(virtualBox, OpenDVDImage(Bstr(argv[2]), emptyUUID, dvdImage.asOutParam())); if (SUCCEEDED(rc) && dvdImage) { /* time to register the image */ CHECK_ERROR(virtualBox, RegisterDVDImage(dvdImage)); } } } if (!dvdImage) { rc = E_FAIL; break; } dvdImage->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(dvdDrive, MountImage(uuid)); } } else if (strcmp(argv[1], "floppyattach") == 0) { if (argc != 3) { errorSyntax(USAGE_CONTROLVM, "Incorrect number of parameters"); rc = E_FAIL; break; } ComPtr floppyDrive; sessionMachine->COMGETTER(FloppyDrive)(floppyDrive.asOutParam()); ASSERT(floppyDrive); /* unmount? */ if (strcmp(argv[2], "none") == 0) { CHECK_ERROR(floppyDrive, Unmount()); } /* host drive? */ else if (strncmp(argv[2], "host:", 5) == 0) { ComPtr host; CHECK_ERROR(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr hostFloppies; CHECK_ERROR(host, COMGETTER(FloppyDrives)(hostFloppies.asOutParam())); ComPtr hostFloppyDrive; rc = hostFloppies->FindByName(Bstr(argv[2] + 5), hostFloppyDrive.asOutParam()); if (!hostFloppyDrive) { errorArgument("Invalid host floppy drive name"); rc = E_FAIL; break; } CHECK_ERROR(floppyDrive, CaptureHostDrive(hostFloppyDrive)); } else { /* first assume it's a UUID */ Guid uuid(argv[2]); ComPtr floppyImage; rc = virtualBox->GetFloppyImage(uuid, floppyImage.asOutParam()); if (FAILED(rc) || !floppyImage) { /* must be a filename */ Guid emptyUUID; CHECK_ERROR(virtualBox, OpenFloppyImage(Bstr(argv[2]), emptyUUID, floppyImage.asOutParam())); if (SUCCEEDED(rc) && floppyImage) { /** @todo first iterate through the collection and try to find the image */ /* time to register the image */ CHECK_ERROR(virtualBox, RegisterFloppyImage(floppyImage)); } } if (!floppyImage) { rc = E_FAIL; break; } floppyImage->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(floppyDrive, MountImage(uuid)); } } else { errorSyntax(USAGE_CONTROLVM, "Invalid parameter '%s'", Utf8Str(argv[1]).raw()); rc = E_FAIL; } } while (0); session->Close(); return SUCCEEDED (rc) ? 0 : 1; } static int handleDiscardState(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; if (argc != 1) { return errorSyntax(USAGE_DISCARDSTATE, "Incorrect number of parameters"); } ComPtr machine; /* assume it's a UUID */ rc = virtualBox->GetMachine(Guid(argv[0]), machine.asOutParam()); if (FAILED(rc) || !machine) { /* must be a name */ CHECK_ERROR(virtualBox, FindMachine(Bstr(argv[0]), machine.asOutParam())); } if (machine) { do { /* we have to open a session for this task */ Guid guid; machine->COMGETTER(Id)(guid.asOutParam()); CHECK_ERROR_BREAK(virtualBox, OpenSession(session, guid)); ComPtr console; CHECK_ERROR_BREAK(session, COMGETTER(Console)(console.asOutParam())); CHECK_ERROR_BREAK(console, DiscardSavedState()); CHECK_ERROR_BREAK(session, Close()); } while (0); } return SUCCEEDED(rc) ? 0 : 1; } static int handleSnapshot(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; /* we need at least a VM and a command */ if (argc < 2) { return errorSyntax(USAGE_SNAPSHOT, "Not enough parameters"); } /* the first argument must be the VM */ ComPtr machine; /* assume it's a UUID */ rc = virtualBox->GetMachine(Guid(argv[0]), machine.asOutParam()); if (FAILED(rc) || !machine) { /* must be a name */ CHECK_ERROR(virtualBox, FindMachine(Bstr(argv[0]), machine.asOutParam())); } if (!machine) return 1; Guid guid; machine->COMGETTER(Id)(guid.asOutParam()); do { /* we have to open a session for this task. First try an existing session */ rc = virtualBox->OpenExistingSession(session, guid); if (FAILED(rc)) CHECK_ERROR_BREAK(virtualBox, OpenSession(session, guid)); ComPtr console; CHECK_ERROR_BREAK(session, COMGETTER(Console)(console.asOutParam())); /* switch based on the command */ if (strcmp(argv[1], "take") == 0) { /* there must be a name */ if (argc < 3) { errorSyntax(USAGE_SNAPSHOT, "Missing snapshot name"); rc = E_FAIL; break; } Bstr name(argv[2]); if ((argc > 3) && ((argc != 5) || (strcmp(argv[3], "-desc") != 0))) { errorSyntax(USAGE_SNAPSHOT, "Incorrect description format"); rc = E_FAIL; break; } Bstr desc; if (argc == 5) desc = argv[4]; ComPtr progress; CHECK_ERROR_BREAK(console, TakeSnapshot(name, desc, progress.asOutParam())); showProgress(progress); progress->COMGETTER(ResultCode)(&rc); if (FAILED(rc)) { com::ProgressErrorInfo info(progress); if (info.isBasicAvailable()) { RTPrintf("Error: failed to take snapshot. Error message: %lS\n", info.getText().raw()); } else { RTPrintf("Error: failed to take snapshot. No error message available!\n"); } } } else if (strcmp(argv[1], "discard") == 0) { /* exactly one parameter: snapshot name */ if (argc != 3) { errorSyntax(USAGE_SNAPSHOT, "Expecting snapshot name only"); rc = E_FAIL; break; } ComPtr snapshot; /* assume it's a UUID */ Guid guid(argv[2]); if (!guid.isEmpty()) { CHECK_ERROR_BREAK(machine, GetSnapshot(guid, snapshot.asOutParam())); } else { /* then it must be a name */ CHECK_ERROR_BREAK(machine, FindSnapshot(Bstr(argv[2]), snapshot.asOutParam())); } snapshot->COMGETTER(Id)(guid.asOutParam()); ComPtr progress; CHECK_ERROR_BREAK(console, DiscardSnapshot(guid, progress.asOutParam())); showProgress(progress); progress->COMGETTER(ResultCode)(&rc); if (FAILED(rc)) { com::ProgressErrorInfo info(progress); if (info.isBasicAvailable()) { RTPrintf("Error: failed to discard snapshot. Error message: %lS\n", info.getText().raw()); } else { RTPrintf("Error: failed to discard snapshot. No error message available!\n"); } } } else if (strcmp(argv[1], "discardcurrent") == 0) { if ( (argc != 3) || ( (strcmp(argv[2], "-state") != 0) && (strcmp(argv[2], "-all") != 0))) { errorSyntax(USAGE_SNAPSHOT, "Invalid parameter '%s'", Utf8Str(argv[2]).raw()); rc = E_FAIL; break; } bool fAll = false; if (strcmp(argv[2], "-all") == 0) fAll = true; ComPtr progress; if (fAll) { CHECK_ERROR_BREAK(console, DiscardCurrentSnapshotAndState(progress.asOutParam())); } else { CHECK_ERROR_BREAK(console, DiscardCurrentState(progress.asOutParam())); } showProgress(progress); progress->COMGETTER(ResultCode)(&rc); if (FAILED(rc)) { com::ProgressErrorInfo info(progress); if (info.isBasicAvailable()) { RTPrintf("Error: failed to discard. Error message: %lS\n", info.getText().raw()); } else { RTPrintf("Error: failed to discard. No error message available!\n"); } } } else if (strcmp(argv[1], "edit") == 0) { if (argc < 3) { errorSyntax(USAGE_SNAPSHOT, "Missing snapshot name"); rc = E_FAIL; break; } ComPtr snapshot; if (strcmp(argv[2], "-current") == 0) { CHECK_ERROR_BREAK(machine, COMGETTER(CurrentSnapshot)(snapshot.asOutParam())); } else { /* assume it's a UUID */ Guid guid(argv[2]); if (!guid.isEmpty()) { CHECK_ERROR_BREAK(machine, GetSnapshot(guid, snapshot.asOutParam())); } else { /* then it must be a name */ CHECK_ERROR_BREAK(machine, FindSnapshot(Bstr(argv[2]), snapshot.asOutParam())); } } /* parse options */ for (int i = 3; i < argc; i++) { if (strcmp(argv[i], "-newname") == 0) { if (argc <= i + 1) { errorArgument("Missing argument to '%s'", argv[i]); rc = E_FAIL; break; } i++; snapshot->COMSETTER(Name)(Bstr(argv[i])); } else if (strcmp(argv[i], "-newdesc") == 0) { if (argc <= i + 1) { errorArgument("Missing argument to '%s'", argv[i]); rc = E_FAIL; break; } i++; snapshot->COMSETTER(Description)(Bstr(argv[i])); } else { errorSyntax(USAGE_SNAPSHOT, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); rc = E_FAIL; break; } } } else if (strcmp(argv[1], "showvminfo") == 0) { /* exactly one parameter: snapshot name */ if (argc != 3) { errorSyntax(USAGE_SNAPSHOT, "Expecting snapshot name only"); rc = E_FAIL; break; } ComPtr snapshot; /* assume it's a UUID */ Guid guid(argv[2]); if (!guid.isEmpty()) { CHECK_ERROR_BREAK(machine, GetSnapshot(guid, snapshot.asOutParam())); } else { /* then it must be a name */ CHECK_ERROR_BREAK(machine, FindSnapshot(Bstr(argv[2]), snapshot.asOutParam())); } /* get the machine of the given snapshot */ ComPtr machine; snapshot->COMGETTER(Machine)(machine.asOutParam()); showVMInfo(virtualBox, machine, console); } else { errorSyntax(USAGE_SNAPSHOT, "Invalid parameter '%s'", Utf8Str(argv[1]).raw()); rc = E_FAIL; } } while (0); session->Close(); return SUCCEEDED(rc) ? 0 : 1; } static int handleShowVDIInfo(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; if (argc != 1) { return errorSyntax(USAGE_SHOWVDIINFO, "Incorrect number of parameters"); } ComPtr hardDisk; ComPtr vdi; Bstr filepath; bool registered = true; /* first guess is that it's a UUID */ Guid uuid(argv[0]); rc = virtualBox->GetHardDisk(uuid, hardDisk.asOutParam()); /* no? then it must be a filename */ if (FAILED (rc)) { filepath = argv[0]; rc = virtualBox->FindVirtualDiskImage(filepath, vdi.asOutParam()); /* no? well, then it's an unregistered image */ if (FAILED (rc)) { registered = false; CHECK_ERROR(virtualBox, OpenVirtualDiskImage(filepath, vdi.asOutParam())); } if (SUCCEEDED (rc)) hardDisk = vdi; } else { vdi = hardDisk; } if (SUCCEEDED(rc) && hardDisk) { hardDisk->COMGETTER(Id)(uuid.asOutParam()); RTPrintf("UUID: %s\n", uuid.toString().raw()); RTPrintf("Registered: %s\n", registered ? "yes" : "no"); /* check for accessibility */ BOOL accessible = FALSE; CHECK_ERROR_RET (hardDisk, COMGETTER(Accessible)(&accessible), 1); RTPrintf("Accessible: %s\n", accessible ? "yes" : "no"); if (accessible) { Bstr description; hardDisk->COMGETTER(Description)(description.asOutParam()); if (description) { RTPrintf("Description: %lS\n", description.raw()); } ULONG64 size; hardDisk->COMGETTER(Size)(&size); RTPrintf("Size: %llu MBytes\n", size); ULONG64 actualSize; hardDisk->COMGETTER(ActualSize)(&actualSize); RTPrintf("Current size on disk: %llu MBytes\n", actualSize >> 20); } else { Bstr err; CHECK_ERROR_RET (hardDisk, COMGETTER(LastAccessError)(err.asOutParam()), 1); RTPrintf("Access Error: %lS\n", err.raw()); } HardDiskType_T type; hardDisk->COMGETTER(Type)(&type); const char *typeStr = "unknown"; switch (type) { case HardDiskType_NormalHardDisk: typeStr = "standard"; break; case HardDiskType_ImmutableHardDisk: typeStr = "immutable"; break; case HardDiskType_WritethroughHardDisk: typeStr = "writethrough"; break; } RTPrintf("Type: %s\n", typeStr); HardDiskStorageType_T storageType; const char *storageTypeStr = "unknown"; hardDisk->COMGETTER(StorageType)(&storageType); switch (storageType) { case HardDiskStorageType_VirtualDiskImage: storageTypeStr = "Virtual Disk Image (VDI)"; break; case HardDiskStorageType_ISCSIHardDisk: storageTypeStr = "iSCSI target"; break; } RTPrintf("Storage type: %s\n", storageTypeStr); if (registered) { hardDisk->COMGETTER(MachineId)(uuid.asOutParam()); RTPrintf("In use by VM: %s\n", uuid ? uuid.toString().raw() : ""); } if (vdi) { /* VDI specific information */ vdi->COMGETTER(FilePath)(filepath.asOutParam()); RTPrintf("Path: %lS\n", filepath.raw()); } else { /* Generic location information */ Bstr loc; hardDisk->COMGETTER(Location)(loc.asOutParam()); RTPrintf("Location: %lS\n", loc.raw()); } } return SUCCEEDED(rc) ? 0 : 1; } static int handleRegisterImage(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; if (argc < 2) { return errorSyntax(USAGE_REGISTERIMAGE, "Not enough parameters"); } Bstr filepath(argv[1]); if (strcmp(argv[0], "disk") == 0) { const char *type = "normal"; /* there can be a type parameter */ if ((argc > 2) && (argc != 4)) { return errorSyntax(USAGE_REGISTERIMAGE, "Incorrect number of parameters"); } if (argc == 4) { if (strcmp(argv[2], "-type") != 0) { return errorSyntax(USAGE_REGISTERIMAGE, "Invalid parameter '%s'", Utf8Str(argv[2]).raw()); } if ( (strcmp(argv[3], "normal") != 0) && (strcmp(argv[3], "immutable") != 0) && (strcmp(argv[3], "writethrough") != 0)) { return errorArgument("Invalid VDI type '%s' specified", Utf8Str(argv[3]).raw()); } type = argv[3]; } ComPtr vdi; CHECK_ERROR(virtualBox, OpenVirtualDiskImage(filepath, vdi.asOutParam())); if (SUCCEEDED(rc) && vdi) { ComPtr hardDisk = vdi; /* change the type if requested */ if (strcmp(type, "normal") == 0) CHECK_ERROR(hardDisk, COMSETTER(Type)(HardDiskType_NormalHardDisk)); else if (strcmp(type, "immutable") == 0) CHECK_ERROR(hardDisk, COMSETTER(Type)(HardDiskType_ImmutableHardDisk)); else if (strcmp(type, "writethrough") == 0) CHECK_ERROR(hardDisk, COMSETTER(Type)(HardDiskType_WritethroughHardDisk)); CHECK_ERROR(virtualBox, RegisterHardDisk(hardDisk)); } } else if (strcmp(argv[0], "dvd") == 0) { ComPtr dvdImage; CHECK_ERROR(virtualBox, OpenDVDImage(filepath, Guid(), dvdImage.asOutParam())); if (SUCCEEDED(rc) && dvdImage) { CHECK_ERROR(virtualBox, RegisterDVDImage(dvdImage)); } } else if (strcmp(argv[0], "floppy") == 0) { ComPtr floppyImage; CHECK_ERROR(virtualBox, OpenFloppyImage(filepath, Guid(), floppyImage.asOutParam())); if (SUCCEEDED(rc) && floppyImage) { CHECK_ERROR(virtualBox, RegisterFloppyImage(floppyImage)); } } else { return errorSyntax(USAGE_REGISTERIMAGE, "Invalid parameter '%s'", Utf8Str(argv[1]).raw()); } return SUCCEEDED(rc) ? 0 : 1; } static int handleUnregisterImage(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; if (argc != 2) { return errorSyntax(USAGE_UNREGISTERIMAGE, "Incorrect number of parameters"); } /* first guess is that it's a UUID */ Guid uuid(argv[1]); if (strcmp(argv[0], "disk") == 0) { ComPtr hardDisk; rc = virtualBox->GetHardDisk(uuid, hardDisk.asOutParam()); /* not a UUID or not registered? Then it must be a filename */ if (!hardDisk) { ComPtr vdi; CHECK_ERROR(virtualBox, FindVirtualDiskImage(Bstr(argv[1]), vdi.asOutParam())); hardDisk = vdi; } if (SUCCEEDED(rc) && hardDisk) { hardDisk->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(virtualBox, UnregisterHardDisk(uuid, hardDisk.asOutParam())); } } else if (strcmp(argv[0], "dvd") == 0) { ComPtr dvdImage; rc = virtualBox->GetDVDImage(uuid, dvdImage.asOutParam()); /* not a UUID or not registered? Then it must be a filename */ if (!dvdImage) { ComPtr dvdColl; virtualBox->COMGETTER(DVDImages)(dvdColl.asOutParam()); CHECK_ERROR(dvdColl, FindByPath(Bstr(argv[1]), dvdImage.asOutParam())); } if (SUCCEEDED(rc) && dvdImage) { dvdImage->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(virtualBox, UnregisterDVDImage(uuid, dvdImage.asOutParam())); } } else if (strcmp(argv[0], "floppy") == 0) { ComPtr floppyImage; rc = virtualBox->GetFloppyImage(uuid, floppyImage.asOutParam()); /* not a UUID or not registered? Then it must be a filename */ if (!floppyImage) { ComPtr floppyColl; virtualBox->COMGETTER(FloppyImages)(floppyColl.asOutParam()); CHECK_ERROR(floppyColl, FindByPath(Bstr(argv[1]), floppyImage.asOutParam())); } if (SUCCEEDED(rc) && floppyImage) { floppyImage->COMGETTER(Id)(uuid.asOutParam()); CHECK_ERROR(virtualBox, UnregisterFloppyImage(uuid, floppyImage.asOutParam())); } } else { return errorSyntax(USAGE_UNREGISTERIMAGE, "Invalid parameter '%s'", Utf8Str(argv[1]).raw()); } return SUCCEEDED(rc) ? 0 : 1; } #ifdef __WIN__ static int handleCreateHostIF(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { if (argc != 1) { return errorSyntax(USAGE_CREATEHOSTIF, "Incorrect number of parameters"); } HRESULT rc = S_OK; do { ComPtr host; CHECK_ERROR_BREAK(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr hostif; ComPtr progress; CHECK_ERROR_BREAK(host, CreateHostNetworkInterface(Bstr(argv[0]), hostif.asOutParam(), progress.asOutParam())); showProgress(progress); HRESULT result; CHECK_ERROR_BREAK(progress, COMGETTER(ResultCode)(&result)); if (FAILED(result)) { com::ProgressErrorInfo info(progress); PRINT_ERROR_INFO(info); rc = result; } } while (0); return SUCCEEDED(rc) ? 0 : 1; } static int handleRemoveHostIF(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { if (argc != 1) { return errorSyntax(USAGE_REMOVEHOSTIF, "Incorrect number of parameters"); } HRESULT rc = S_OK; do { ComPtr host; CHECK_ERROR_BREAK(virtualBox, COMGETTER(Host)(host.asOutParam())); ComPtr hostif; /* first guess is that it's a UUID */ Guid uuid(argv[0]); if (uuid.isEmpty()) { /* not a valid UUID, search for it */ ComPtr coll; CHECK_ERROR_BREAK(host, COMGETTER(NetworkInterfaces)(coll.asOutParam())); CHECK_ERROR_BREAK(coll, FindByName(Bstr(argv[0]), hostif.asOutParam())); CHECK_ERROR_BREAK(hostif, COMGETTER(Id)(uuid.asOutParam())); } ComPtr progress; CHECK_ERROR_BREAK(host, RemoveHostNetworkInterface(uuid, hostif.asOutParam(), progress.asOutParam())); showProgress(progress); HRESULT result; CHECK_ERROR_BREAK(progress, COMGETTER(ResultCode)(&result)); if (FAILED(result)) { com::ProgressErrorInfo info(progress); PRINT_ERROR_INFO(info); rc = result; } } while (0); return SUCCEEDED(rc) ? 0 : 1; } #endif /* __WIN__ */ static int handleGetExtraData(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc = S_OK; if (argc != 2) { return errorSyntax(USAGE_GETEXTRADATA, "Incorrect number of parameters"); } /* global data? */ if (strcmp(argv[0], "global") == 0) { /* enumeration? */ if (strcmp(argv[1], "enumerate") == 0) { Bstr extraDataKey; do { Bstr nextExtraDataKey; Bstr nextExtraDataValue; HRESULT rcEnum = virtualBox->GetNextExtraDataKey(extraDataKey, nextExtraDataKey.asOutParam(), nextExtraDataValue.asOutParam()); extraDataKey = nextExtraDataKey; if (SUCCEEDED(rcEnum) && extraDataKey) { RTPrintf("Key: %lS, Value: %lS\n", nextExtraDataKey.raw(), nextExtraDataValue.raw()); } } while (extraDataKey); } else { Bstr value; CHECK_ERROR(virtualBox, GetExtraData(Bstr(argv[1]), value.asOutParam())); if (value) RTPrintf("Value: %lS\n", value.raw()); else RTPrintf("No value set!\n"); } } else { ComPtr machine; /* assume it's a UUID */ rc = virtualBox->GetMachine(Guid(argv[0]), machine.asOutParam()); if (FAILED(rc) || !machine) { /* must be a name */ CHECK_ERROR(virtualBox, FindMachine(Bstr(argv[0]), machine.asOutParam())); } if (machine) { /* enumeration? */ if (strcmp(argv[1], "enumerate") == 0) { Bstr extraDataKey; do { Bstr nextExtraDataKey; Bstr nextExtraDataValue; HRESULT rcEnum = machine->GetNextExtraDataKey(extraDataKey, nextExtraDataKey.asOutParam(), nextExtraDataValue.asOutParam()); extraDataKey = nextExtraDataKey; if (SUCCEEDED(rcEnum) && extraDataKey) { RTPrintf("Key: %lS, Value: %lS\n", nextExtraDataKey.raw(), nextExtraDataValue.raw()); } } while (extraDataKey); } else { Bstr value; CHECK_ERROR(machine, GetExtraData(Bstr(argv[1]), value.asOutParam())); if (value) RTPrintf("Value: %lS\n", value.raw()); else RTPrintf("No value set!\n"); } } } return SUCCEEDED(rc) ? 0 : 1; } static int handleSetExtraData(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc = S_OK; if (argc < 2) { return errorSyntax(USAGE_SETEXTRADATA, "Not enough parameters"); } /* global data? */ if (strcmp(argv[0], "global") == 0) { if (argc < 3) CHECK_ERROR(virtualBox, SetExtraData(Bstr(argv[1]), NULL)); else if (argc == 3) CHECK_ERROR(virtualBox, SetExtraData(Bstr(argv[1]), Bstr(argv[2]))); else return errorSyntax(USAGE_SETEXTRADATA, "Too many parameters"); } else { ComPtr machine; /* assume it's a UUID */ rc = virtualBox->GetMachine(Guid(argv[0]), machine.asOutParam()); if (FAILED(rc) || !machine) { /* must be a name */ CHECK_ERROR(virtualBox, FindMachine(Bstr(argv[0]), machine.asOutParam())); } if (machine) { if (argc < 3) CHECK_ERROR(machine, SetExtraData(Bstr(argv[1]), NULL)); else if (argc == 3) CHECK_ERROR(machine, SetExtraData(Bstr(argv[1]), Bstr(argv[2]))); else return errorSyntax(USAGE_SETEXTRADATA, "Too many parameters"); } } return SUCCEEDED(rc) ? 0 : 1; } static int handleSetProperty(int argc, char *argv[], ComPtr virtualBox, ComPtr session) { HRESULT rc; /* there must be two arguments: property name and value */ if (argc != 2) { return errorSyntax(USAGE_SETPROPERTY, "Incorrect number of parameters"); } ComPtr systemProperties; virtualBox->COMGETTER(SystemProperties)(systemProperties.asOutParam()); if (strcmp(argv[0], "vdifolder") == 0) { /* reset to default? */ if (strcmp(argv[1], "default") == 0) { CHECK_ERROR(systemProperties, COMSETTER(DefaultVDIFolder)(NULL)); } else { CHECK_ERROR(systemProperties, COMSETTER(DefaultVDIFolder)(Bstr(argv[1]))); } } else if (strcmp(argv[0], "machinefolder") == 0) { /* reset to default? */ if (strcmp(argv[1], "default") == 0) { CHECK_ERROR(systemProperties, COMSETTER(DefaultMachineFolder)(NULL)); } else { CHECK_ERROR(systemProperties, COMSETTER(DefaultMachineFolder)(Bstr(argv[1]))); } } else if (strcmp(argv[0], "vrdpauthlibrary") == 0) { /* reset to default? */ if (strcmp(argv[1], "default") == 0) { CHECK_ERROR(systemProperties, COMSETTER(RemoteDisplayAuthLibrary)(NULL)); } else { CHECK_ERROR(systemProperties, COMSETTER(RemoteDisplayAuthLibrary)(Bstr(argv[1]))); } } else if (strcmp(argv[0], "hwvirtexenabled") == 0) { if (strcmp(argv[1], "yes") == 0) CHECK_ERROR(systemProperties, COMSETTER(HWVirtExEnabled)(TRUE)); else if (strcmp(argv[1], "no") == 0) CHECK_ERROR(systemProperties, COMSETTER(HWVirtExEnabled)(FALSE)); else return errorArgument("Invalid value '%s' for hardware virtualization extension flag", argv[1]); } else { return errorSyntax(USAGE_SETPROPERTY, "Invalid parameter '%s'", Utf8Str(argv[0]).raw()); } return SUCCEEDED(rc) ? 0 : 1; } static int handleUSBFilter (int argc, char *argv[], ComPtr aVirtualBox, ComPtr aSession) { HRESULT rc = S_OK; USBFilterCmd cmd; /* at least: 0: command, 1: index, 2: -target, 3: */ if (argc < 4) { return errorSyntax(USAGE_USBFILTER, "Not enough parameters"); } /* which command? */ cmd.mAction = USBFilterCmd::Invalid; if (strcmp (argv [0], "add") == 0) cmd.mAction = USBFilterCmd::Add; else if (strcmp (argv [0], "modify") == 0) cmd.mAction = USBFilterCmd::Modify; else if (strcmp (argv [0], "remove") == 0) cmd.mAction = USBFilterCmd::Remove; if (cmd.mAction == USBFilterCmd::Invalid) { return errorSyntax(USAGE_USBFILTER, "Invalid parameter '%s'", Utf8Str(argv[0]).raw()); } /* which index? */ char *endptr = NULL; cmd.mIndex = strtoul (argv[1], &endptr, 10); if (!endptr || *endptr) { return errorSyntax(USAGE_USBFILTER, "Invalid index '%s'", argv[1]); } switch (cmd.mAction) { case USBFilterCmd::Add: case USBFilterCmd::Modify: { /* at least: 0: command, 1: index, 2: -target, 3: , 4: -name, 5: */ if (argc < 6) { if (cmd.mAction == USBFilterCmd::Add) { return errorSyntax(USAGE_USBFILTER_ADD, "Not enough parameters"); } else { return errorSyntax(USAGE_USBFILTER_MODIFY, "Not enough parameters"); } } // set Active to true by default // (assuming that the user sets up all necessary attributes // at once and wants the filter to be active immediately) if (cmd.mAction == USBFilterCmd::Add) cmd.mFilter.mActive = true; for (int i = 2; i < argc; i++) { if (strcmp(argv [i], "-target") == 0) { if (argc <= i + 1 || !*argv[i+1]) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; if (strcmp (argv [i], "global") == 0) cmd.mGlobal = true; else { /* assume it's a UUID of a machine */ rc = aVirtualBox->GetMachine(Guid(argv[i]), cmd.mMachine.asOutParam()); if (FAILED(rc) || !cmd.mMachine) { /* must be a name */ CHECK_ERROR_RET(aVirtualBox, FindMachine(Bstr(argv[i]), cmd.mMachine.asOutParam()), 1); } } } else if (strcmp(argv [i], "-name") == 0) { if (argc <= i + 1 || !*argv[i+1]) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; cmd.mFilter.mName = argv [i]; } else if (strcmp(argv [i], "-active") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; if (strcmp (argv [i], "yes") == 0) cmd.mFilter.mActive = true; else if (strcmp (argv [i], "no") == 0) cmd.mFilter.mActive = false; else { return errorArgument("Invalid -active argument '%s'", Utf8Str(argv[i]).raw()); } } else if (strcmp(argv [i], "-vendorid") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; cmd.mFilter.mVendorId = argv [i]; } else if (strcmp(argv [i], "-productid") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; cmd.mFilter.mProductId = argv [i]; } else if (strcmp(argv [i], "-revision") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; cmd.mFilter.mRevision = argv [i]; } else if (strcmp(argv [i], "-manufacturer") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; cmd.mFilter.mManufacturer = argv [i]; } else if (strcmp(argv [i], "-product") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; cmd.mFilter.mProduct = argv [i]; } else if (strcmp(argv [i], "-remote") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; cmd.mFilter.mRemote = argv[i]; } else if (strcmp(argv [i], "-serialnumber") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; cmd.mFilter.mSerialNumber = argv [i]; } else if (strcmp(argv [i], "-action") == 0) { if (argc <= i + 1) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; if (strcmp (argv [i], "ignore") == 0) cmd.mFilter.mAction = USBDeviceFilterAction_USBDeviceFilterIgnore; else if (strcmp (argv [i], "hold") == 0) cmd.mFilter.mAction = USBDeviceFilterAction_USBDeviceFilterHold; else { return errorArgument("Invalid USB filter action '%s'", Utf8Str(argv[i]).raw()); } } } if (cmd.mAction == USBFilterCmd::Add) { // mandatory/forbidden options if ( cmd.mFilter.mName.isEmpty() || ( cmd.mGlobal && cmd.mFilter.mAction == USBDeviceFilterAction_InvalidUSBDeviceFilterAction ) || ( !cmd.mGlobal && !cmd.mMachine) || ( cmd.mGlobal && cmd.mFilter.mRemote) ) { return errorSyntax(USAGE_USBFILTER_ADD, "Mandatory options not supplied"); } } break; } case USBFilterCmd::Remove: { /* at least: 0: command, 1: index, 2: -target, 3: */ if (argc < 4) { return errorSyntax(USAGE_USBFILTER_REMOVE, "Not enough parameters"); } for (int i = 2; i < argc; i++) { if (strcmp(argv [i], "-target") == 0) { if (argc <= i + 1 || !*argv[i+1]) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; if (strcmp (argv [i], "global") == 0) cmd.mGlobal = true; else { /* assume it's a UUID of a machine */ rc = aVirtualBox->GetMachine(Guid(argv[i]), cmd.mMachine.asOutParam()); if (FAILED(rc) || !cmd.mMachine) { /* must be a name */ CHECK_ERROR_RET(aVirtualBox, FindMachine(Bstr(argv[i]), cmd.mMachine.asOutParam()), 1); } } } } // mandatory options if (!cmd.mGlobal && !cmd.mMachine) { return errorSyntax(USAGE_USBFILTER_REMOVE, "Mandatory options not supplied"); } break; } default: break; } USBFilterCmd::USBFilter &f = cmd.mFilter; ComPtr host; ComPtr ctl; if (cmd.mGlobal) CHECK_ERROR_RET (aVirtualBox, COMGETTER(Host) (host.asOutParam()), 1); else { Guid uuid; cmd.mMachine->COMGETTER(Id)(uuid.asOutParam()); /* open a session for the VM */ CHECK_ERROR_RET (aVirtualBox, OpenSession(aSession, uuid), 1); /* get the mutable session machine */ aSession->COMGETTER(Machine)(cmd.mMachine.asOutParam()); /* and get the USB controller */ CHECK_ERROR_RET (cmd.mMachine, COMGETTER(USBController) (ctl.asOutParam()), 1); } switch (cmd.mAction) { case USBFilterCmd::Add: { if (cmd.mGlobal) { ComPtr flt; CHECK_ERROR_BREAK (host, CreateUSBDeviceFilter (f.mName, flt.asOutParam())); if (!f.mActive.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Active) (f.mActive)); if (!f.mVendorId.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(VendorId) (f.mVendorId.setNullIfEmpty())); if (!f.mProductId.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(ProductId) (f.mProductId.setNullIfEmpty())); if (!f.mRevision.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Revision) (f.mRevision.setNullIfEmpty())); if (!f.mManufacturer.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Manufacturer) (f.mManufacturer.setNullIfEmpty())); if (!f.mSerialNumber.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(SerialNumber) (f.mSerialNumber.setNullIfEmpty())); if (f.mAction != USBDeviceFilterAction_InvalidUSBDeviceFilterAction) CHECK_ERROR_BREAK (flt, COMSETTER(Action) (f.mAction)); CHECK_ERROR_BREAK (host, InsertUSBDeviceFilter (cmd.mIndex, flt)); } else { ComPtr flt; CHECK_ERROR_BREAK (ctl, CreateDeviceFilter (f.mName, flt.asOutParam())); if (!f.mActive.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Active) (f.mActive)); if (!f.mVendorId.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(VendorId) (f.mVendorId.setNullIfEmpty())); if (!f.mProductId.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(ProductId) (f.mProductId.setNullIfEmpty())); if (!f.mRevision.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Revision) (f.mRevision.setNullIfEmpty())); if (!f.mManufacturer.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Manufacturer) (f.mManufacturer.setNullIfEmpty())); if (!f.mRemote.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Remote) (f.mRemote.setNullIfEmpty())); if (!f.mSerialNumber.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(SerialNumber) (f.mSerialNumber.setNullIfEmpty())); CHECK_ERROR_BREAK (ctl, InsertDeviceFilter (cmd.mIndex, flt)); } break; } case USBFilterCmd::Modify: { if (cmd.mGlobal) { ComPtr coll; CHECK_ERROR_BREAK (host, COMGETTER(USBDeviceFilters) (coll.asOutParam())); ComPtr flt; CHECK_ERROR_BREAK (coll, GetItemAt (cmd.mIndex, flt.asOutParam())); if (!f.mName.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Name) (f.mName.setNullIfEmpty())); if (!f.mActive.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Active) (f.mActive)); if (!f.mVendorId.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(VendorId) (f.mVendorId.setNullIfEmpty())); if (!f.mProductId.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(ProductId) (f.mProductId.setNullIfEmpty())); if (!f.mRevision.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Revision) (f.mRevision.setNullIfEmpty())); if (!f.mManufacturer.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Manufacturer) (f.mManufacturer.setNullIfEmpty())); if (!f.mSerialNumber.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(SerialNumber) (f.mSerialNumber.setNullIfEmpty())); if (f.mAction != USBDeviceFilterAction_InvalidUSBDeviceFilterAction) CHECK_ERROR_BREAK (flt, COMSETTER(Action) (f.mAction)); } else { ComPtr coll; CHECK_ERROR_BREAK (ctl, COMGETTER(DeviceFilters) (coll.asOutParam())); ComPtr flt; CHECK_ERROR_BREAK (coll, GetItemAt (cmd.mIndex, flt.asOutParam())); if (!f.mName.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Name) (f.mName.setNullIfEmpty())); if (!f.mActive.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Active) (f.mActive)); if (!f.mVendorId.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(VendorId) (f.mVendorId.setNullIfEmpty())); if (!f.mProductId.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(ProductId) (f.mProductId.setNullIfEmpty())); if (!f.mRevision.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Revision) (f.mRevision.setNullIfEmpty())); if (!f.mManufacturer.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Manufacturer) (f.mManufacturer.setNullIfEmpty())); if (!f.mRemote.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(Remote) (f.mRemote.setNullIfEmpty())); if (!f.mSerialNumber.isNull()) CHECK_ERROR_BREAK (flt, COMSETTER(SerialNumber) (f.mSerialNumber.setNullIfEmpty())); } break; } case USBFilterCmd::Remove: { if (cmd.mGlobal) { ComPtr flt; CHECK_ERROR_BREAK (host, RemoveUSBDeviceFilter (cmd.mIndex, flt.asOutParam())); } else { ComPtr flt; CHECK_ERROR_BREAK (ctl, RemoveDeviceFilter (cmd.mIndex, flt.asOutParam())); } break; } default: break; } if (cmd.mMachine) { /* commit and close the session */ CHECK_ERROR(cmd.mMachine, SaveSettings()); aSession->Close(); } return SUCCEEDED (rc) ? 0 : 1; } static int handleSharedFolder (int argc, char *argv[], ComPtr aVirtualBox, ComPtr aSession) { HRESULT rc; /* we need at least a command and target */ if (argc < 2) { return errorSyntax(USAGE_SHAREDFOLDER, "Not enough parameters"); } ComPtr machine; /* assume it's a UUID */ rc = aVirtualBox->GetMachine(Guid(argv[1]), machine.asOutParam()); if (FAILED(rc) || !machine) { /* must be a name */ CHECK_ERROR(aVirtualBox, FindMachine(Bstr(argv[1]), machine.asOutParam())); } if (!machine) return 1; Guid uuid; machine->COMGETTER(Id)(uuid.asOutParam()); if (strcmp(argv[0], "add") == 0) { /* we need at least four more parameters */ if (argc < 5) { return errorSyntax(USAGE_SHAREDFOLDER_ADD, "Not enough parameters"); } char *name = NULL; char *hostpath = NULL; bool fTransient = false; for (int i = 2; i < argc; i++) { if (strcmp(argv[i], "-name") == 0) { if (argc <= i + 1 || !*argv[i+1]) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; name = argv[i]; } else if (strcmp(argv[i], "-hostpath") == 0) { if (argc <= i + 1 || !*argv[i+1]) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; hostpath = argv[i]; } else if (strcmp(argv[i], "-transient") == 0) { fTransient = true; } else { return errorSyntax(USAGE_SHAREDFOLDER_ADD, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); } } /* required arguments */ if (!name || !hostpath) { return errorSyntax(USAGE_SHAREDFOLDER_ADD, "Parameters -name and -hostpath are required"); } if (fTransient) { ComPtr console; /* open an existing session for the VM */ CHECK_ERROR_RET(aVirtualBox, OpenExistingSession (aSession, uuid), 1); /* get the session machine */ CHECK_ERROR_RET(aSession, COMGETTER(Machine)(machine.asOutParam()), 1); /* get the session console */ CHECK_ERROR_RET(aSession, COMGETTER(Console)(console.asOutParam()), 1); CHECK_ERROR(console, CreateSharedFolder(Bstr(name), Bstr(hostpath))); if (console) aSession->Close(); } else { /* open a session for the VM */ CHECK_ERROR_RET (aVirtualBox, OpenSession(aSession, uuid), 1); /* get the mutable session machine */ aSession->COMGETTER(Machine)(machine.asOutParam()); CHECK_ERROR(machine, CreateSharedFolder(Bstr(name), Bstr(hostpath))); if (SUCCEEDED(rc)) CHECK_ERROR(machine, SaveSettings()); aSession->Close(); } } else if (strcmp(argv[0], "remove") == 0) { /* we need at least two more parameters */ if (argc < 3) { return errorSyntax(USAGE_SHAREDFOLDER_REMOVE, "Not enough parameters"); } char *name = NULL; bool fTransient = false; for (int i = 2; i < argc; i++) { if (strcmp(argv[i], "-name") == 0) { if (argc <= i + 1 || !*argv[i+1]) { return errorArgument("Missing argument to '%s'", argv[i]); } i++; name = argv[i]; } else if (strcmp(argv[i], "-transient") == 0) { fTransient = true; } else { return errorSyntax(USAGE_SHAREDFOLDER_REMOVE, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); } } /* required arguments */ if (!name) { return errorSyntax(USAGE_SHAREDFOLDER_REMOVE, "Parameter -name is required"); } if (fTransient) { ComPtr console; /* open an existing session for the VM */ CHECK_ERROR_RET(aVirtualBox, OpenExistingSession (aSession, uuid), 1); /* get the session machine */ CHECK_ERROR_RET(aSession, COMGETTER(Machine)(machine.asOutParam()), 1); /* get the session console */ CHECK_ERROR_RET(aSession, COMGETTER(Console)(console.asOutParam()), 1); CHECK_ERROR(console, RemoveSharedFolder(Bstr(name))); if (console) aSession->Close(); } else { /* open a session for the VM */ CHECK_ERROR_RET (aVirtualBox, OpenSession(aSession, uuid), 1); /* get the mutable session machine */ aSession->COMGETTER(Machine)(machine.asOutParam()); CHECK_ERROR(machine, RemoveSharedFolder(Bstr(name))); /* commit and close the session */ CHECK_ERROR(machine, SaveSettings()); aSession->Close(); } } else { return errorSyntax(USAGE_SETPROPERTY, "Invalid parameter '%s'", Utf8Str(argv[0]).raw()); } return 0; } enum HUSPD { HUSPD_DryRun, HUSPD_Apply, HUSPD_ApplyNoBackup }; static int handleUpdateSettings_processFile (const char *filePath, HUSPD mode) { RTPrintf ("%s\n", filePath); CFGHANDLE config = 0; char *errMsg = NULL; int vrc = CFGLDRLoad (&config, filePath, NIL_RTFILE, NULL, false, NULL, //cfgLdrEntityResolver, &errMsg); if (VBOX_SUCCESS (vrc)) { CFGNODE vbox = 0; CFGLDRGetNode (config, "VirtualBox", 0, &vbox); Bstr version; CFGLDRQueryBSTR (vbox, "version", version.asOutParam()); CFGLDRReleaseNode (vbox); RTPrintf (" current version : %ls\n", version.raw()); /// @todo (dmik) use cfgLdrEntityResolver later vrc = CFGLDRTransform (config, "SettingsConverter.xsl", NULL, &errMsg); if (VBOX_SUCCESS (vrc)) { CFGLDRGetNode (config, "VirtualBox", 0, &vbox); CFGLDRQueryBSTR (vbox, "version", version.asOutParam()); CFGLDRReleaseNode (vbox); RTPrintf (" new version : %ls\n\n", version.raw()); if (mode != HUSPD_DryRun) { if (mode != HUSPD_ApplyNoBackup) { Utf8StrFmt filePathBak ("%s.bak", filePath); vrc = RTFileCopy (filePath, filePathBak); if (VBOX_FAILURE (vrc)) { RTPrintf ("Error copying '%s' to '%s' (%Vrc)\n", filePath, filePathBak.raw(), vrc); } } if (VBOX_SUCCESS (vrc)) { vrc = CFGLDRSave (config, &errMsg); if (VBOX_FAILURE (vrc)) { RTPrintf ("Error saving the settings file '%s' (%Vrc)%s%s\n", filePath, vrc, errMsg ? "\n" : "", errMsg ? errMsg : ""); } } } } else { RTPrintf ("Could not convert the settings file '%s' (%Vrc)%s%s\n", filePath, vrc, errMsg ? "\n" : "", errMsg ? errMsg : ""); } CFGLDRFree (config); } else { RTPrintf ("Error loading the settings file '%s' (%Vrc)%s%s\n", filePath, vrc, errMsg ? "\n" : "", errMsg ? errMsg : ""); } if (errMsg) RTStrFree (errMsg); return vrc; } static int handleUpdateSettings_processDir (const char *dirPath, HUSPD mode, bool skipInvalid) { PRTDIR dir; int vrc = RTDirOpen (&dir, dirPath); if (VBOX_FAILURE (vrc)) { return vrc; } RTDIRENTRYEX entry; while (VBOX_SUCCESS (vrc)) { vrc = RTDirReadEx (dir, &entry, NULL, RTFSOBJATTRADD_UNIX); if (VBOX_FAILURE (vrc)) { if (vrc == VERR_NO_MORE_FILES) vrc = VINF_SUCCESS; else RTPrintf ("Error reading directory '%s' (%Vrc)\n", dirPath, vrc); break; } if (RTFS_IS_DIRECTORY (entry.Info.Attr.fMode)) { if (entry.szName[0] == '.' && (entry.szName[1] == 0 || (entry.szName[1] == '.' && entry.szName[2] == 0))) continue; vrc = handleUpdateSettings_processDir ( Utf8StrFmt ("%s%c%s", dirPath, RTPATH_DELIMITER, entry.szName), mode, skipInvalid); if (VBOX_FAILURE (vrc)) break; continue; } else if (RTFS_IS_FILE (entry.Info.Attr.fMode)) { const char *ext = RTPathExt (entry.szName); if (!ext || strcmp (ext, ".xml") != 0) continue; } else continue; Utf8Str filePath = Utf8StrFmt ("%s%c%s", dirPath, RTPATH_DELIMITER, entry.szName); vrc = handleUpdateSettings_processFile (filePath, mode); if (skipInvalid) vrc = VINF_SUCCESS; } RTDirClose (dir); return vrc; } static int handleUpdateSettings (int argc, char *argv[]) { const char *dirOrFilePath = NULL; bool apply = false; bool nobackup = false; bool skipinvalid = false; for (int i = 0; i < argc; i++) { if (i == 0 && argv[i][0] != '-') { dirOrFilePath = argv[i]; } else if (argv[i][0] == '-') { if (strcmp (&argv[i][1], "apply") == 0) apply = true; else if (strcmp (&argv[i][1], "nobackup") == 0) nobackup = true; else if (strcmp (&argv[i][1], "skipinvalid") == 0) skipinvalid = true; else { return errorSyntax(USAGE_SETPROPERTY, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); } } else { return errorSyntax(USAGE_SETPROPERTY, "Invalid parameter '%s'", Utf8Str(argv[i]).raw()); } } HUSPD mode = HUSPD_DryRun; if (apply) mode = nobackup ? HUSPD_ApplyNoBackup : HUSPD_Apply; int vrc = CFGLDRInitialize(); if (VBOX_FAILURE (vrc)) { RTPrintf ("Could not initialize XML subsystem (%Vrc)\n", vrc); return 1; } if (dirOrFilePath) { if (RTDirExists (dirOrFilePath)) { char fullPath [RTPATH_MAX]; vrc = RTPathReal (dirOrFilePath, fullPath, RTPATH_MAX); if (VBOX_FAILURE (vrc)) { RTPrintf ("Invalid directory path '%s' (%Vrc)\n", dirOrFilePath, vrc); return 1; } RTPrintf ("Updating settings files in the following directory:\n" "\n %s\n\n", fullPath); vrc = handleUpdateSettings_processDir (dirOrFilePath, mode, skipinvalid); } else { vrc = handleUpdateSettings_processFile (dirOrFilePath, mode); } } else { // check if an alternative VBox Home directory is set Utf8Str homeDir = getenv ("VBOX_USER_HOME"); if (!homeDir) { // compose the config directory (full path) char home [RTPATH_MAX]; RTPathUserHome (home, RTPATH_MAX); homeDir = Utf8StrFmt ("%s%c%s", home, RTPATH_DELIMITER, ".VirtualBox"); } RTPrintf ("Updating settings files in the following VirtualBox Home Directory:\n" "\n %s\n\n", homeDir.raw()); vrc = handleUpdateSettings_processDir (homeDir, mode, skipinvalid); } if (mode == HUSPD_DryRun) { RTPrintf ("NOTE: No actual changes to the setting files were made.\n" " Repeat the command with the -apply option supplied.\n"); } CFGLDRShutdown(); return VBOX_SUCCESS (vrc) ? 0 : 1; } // main /////////////////////////////////////////////////////////////////////////////// int main(int argc, char *argv[]) { /* * Before we do anything, init the runtime without loading * the support driver. */ RTR3Init(false); RTPrintf("VirtualBox Command Line Management Interface Version %s\n" "(C) 2005-2007 InnoTek Systemberatung GmbH\n" "All rights reserved.\n" "\n", VBOX_VERSION_STRING); /* shortcut for no parameters or help */ if ( (argc < 2) || (strcmp(argv[1], "help") == 0) || (strcmp(argv[1], "-?") == 0) || (strcmp(argv[1], "-h") == 0) || (strcmp(argv[1], "-help") == 0) || (strcmp(argv[1], "--help") == 0)) { printUsage(USAGE_ALL); return 0; } /* Special option to dump really all commands, even the ones not * understood on this platform. */ if (argc == 2 && strcmp(argv[1], "-dumpopts") == 0) { printUsage(USAGE_DUMPOPTS); return 0; } HRESULT rc; CHECK_RC_RET (com::Initialize()); /* * The input is in the host OS'es codepage (NT guarantees ACP). * For VBox we use UTF-8 and convert to UCS-2 when calling (XP)COM APIs. * For simplicity, just convert the argv[] array here. */ for (int i = 2; i < argc; i++) { char *converted; RTStrCurrentCPToUtf8(&converted, argv[i]); argv[i] = converted; } do { // scopes all the stuff till shutdown //////////////////////////////////////////////////////////////////////////// /* update settings command (no VirtualBox instantiation!) */ if (argc >= 2 && (strcmp(argv[1], "updatesettings") == 0)) { rc = handleUpdateSettings(argc - 2, argv + 2); break; } /* convertdd: does not need a VirtualBox instantiation) */ if (argc >= 2 && (strcmp(argv[1], "convertdd") == 0)) { rc = handleConvertDDImage(argc - 2, argv + 2); break; } ComPtr virtualBox; ComPtr session; rc = virtualBox.createLocalObject (CLSID_VirtualBox); if (FAILED(rc)) { RTPrintf ("[!] Failed to create the VirtualBox object!\n"); PRINT_RC_MESSAGE (rc); com::ErrorInfo info; if (!info.isFullAvailable() && !info.isBasicAvailable()) RTPrintf ("[!] Most likely, the VirtualBox COM server is not running " "or failed to start.\n"); else PRINT_ERROR_INFO (info); break; } CHECK_RC_BREAK (session.createInprocObject (CLSID_Session)); /* create the event queue * (here it is necessary only to process remaining XPCOM/IPC events * after the session is closed) */ EventQueue eventQ; /* * All registered command handlers */ struct { const char *command; PFNHANDLER handler; } commandHandlers[] = { { "internalcommands", handleInternalCommands }, { "list", handleList }, { "showvminfo", handleShowVMInfo }, { "registervm", handleRegisterVM }, { "unregistervm", handleUnregisterVM }, { "createvdi", handleCreateVDI }, { "modifyvdi", handleModifyVDI }, { "addiscsidisk", handleAddiSCSIDisk }, { "createvm", handleCreateVM }, { "modifyvm", handleModifyVM }, { "clonevdi", handleCloneVDI }, { "startvm", handleStartVM }, { "controlvm", handleControlVM }, { "discardstate", handleDiscardState }, { "snapshot", handleSnapshot }, { "registerimage", handleRegisterImage }, { "unregisterimage", handleUnregisterImage }, { "showvdiinfo", handleShowVDIInfo }, #ifdef __WIN__ { "createhostif", handleCreateHostIF }, { "removehostif", handleRemoveHostIF }, #endif { "getextradata", handleGetExtraData }, { "setextradata", handleSetExtraData }, { "setproperty", handleSetProperty }, { "usbfilter", handleUSBFilter }, { "sharedfolder", handleSharedFolder }, { NULL, NULL } }; int commandIndex; for (commandIndex = 0; commandHandlers[commandIndex].command != NULL; commandIndex++) { if (strcmp(commandHandlers[commandIndex].command, argv[1]) == 0) { rc = commandHandlers[commandIndex].handler(argc - 2, &argv[2], virtualBox, session); break; } } if (!commandHandlers[commandIndex].command) { rc = errorSyntax(USAGE_ALL, "Invalid command '%s'", Utf8Str(argv[1]).raw()); } // end "all-stuff" scope //////////////////////////////////////////////////////////////////////////// } while (0); com::Shutdown(); /* * Free converted argument vector */ for (int i = 2; i < argc; i++) { RTStrFree(argv[i]); } return rc; }