/** @file * * VirtualBox COM class implementation */ /* * Copyright (C) 2006-2011 Oracle Corporation * * 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 (GPL) 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. */ #include "MediumAttachmentImpl.h" #include "MachineImpl.h" #include "MediumImpl.h" #include "Global.h" #include "AutoStateDep.h" #include "AutoCaller.h" #include "Logging.h" #include //////////////////////////////////////////////////////////////////////////////// // // private member data definition // //////////////////////////////////////////////////////////////////////////////// struct BackupableMediumAttachmentData { BackupableMediumAttachmentData() : lPort(0), lDevice(0), type(DeviceType_Null), fPassthrough(false), fTempEject(false), fNonRotational(false), fDiscard(false), fImplicit(false) { } ComObjPtr pMedium; /* Since MediumAttachment is not a first class citizen when it * comes to managing settings, having a reference to the storage * controller will not work - when settings are changed it will point * to the old, uninitialized instance. Changing this requires * substantial changes to MediumImpl.cpp. */ const Bstr bstrControllerName; /* Same counts for the assigned bandwidth group */ Utf8Str strBandwidthGroup; const LONG lPort; const LONG lDevice; const DeviceType_T type; bool fPassthrough; bool fTempEject; bool fNonRotational; bool fDiscard; bool fImplicit; }; struct MediumAttachment::Data { Data(Machine * const aMachine = NULL) : pMachine(aMachine), fIsEjected(false) { } /** Reference to Machine object, for checking mutable state. */ Machine * const pMachine; /* later: const ComObjPtr mPeer; */ bool fIsEjected; Backupable bd; }; // constructor / destructor ///////////////////////////////////////////////////////////////////////////// HRESULT MediumAttachment::FinalConstruct() { LogFlowThisFunc(("\n")); return BaseFinalConstruct(); } void MediumAttachment::FinalRelease() { LogFlowThisFuncEnter(); uninit(); BaseFinalRelease(); LogFlowThisFuncLeave(); } // public initializer/uninitializer for internal purposes only ///////////////////////////////////////////////////////////////////////////// /** * Initializes the medium attachment object. * * @param aParent Machine object. * @param aMedium Medium object. * @param aController Controller the hard disk is attached to. * @param aPort Port number. * @param aDevice Device number on the port. * @param aPassthrough Whether accesses are directly passed to the host drive. * @param aBandwidthLimit Bandwidth limit in Mbps */ HRESULT MediumAttachment::init(Machine *aParent, Medium *aMedium, const Bstr &aControllerName, LONG aPort, LONG aDevice, DeviceType_T aType, bool aImplicit, bool aPassthrough, bool aTempEject, bool aNonRotational, bool aDiscard, const Utf8Str &strBandwidthGroup) { LogFlowThisFuncEnter(); LogFlowThisFunc(("aParent=%p aMedium=%p aControllerName=%ls aPort=%d aDevice=%d aType=%d aImplicit=%d aPassthrough=%d aTempEject=%d aNonRotational=%d strBandwithGroup=%s\n", aParent, aMedium, aControllerName.raw(), aPort, aDevice, aType, aImplicit, aPassthrough, aTempEject, aNonRotational, strBandwidthGroup.c_str())); if (aType == DeviceType_HardDisk) AssertReturn(aMedium, E_INVALIDARG); /* Enclose the state transition NotReady->InInit->Ready */ AutoInitSpan autoInitSpan(this); AssertReturn(autoInitSpan.isOk(), E_FAIL); m = new Data(); unconst(m->pMachine) = aParent; m->bd.allocate(); m->bd->pMedium = aMedium; unconst(m->bd->strBandwidthGroup) = strBandwidthGroup; unconst(m->bd->bstrControllerName) = aControllerName; unconst(m->bd->lPort) = aPort; unconst(m->bd->lDevice) = aDevice; unconst(m->bd->type) = aType; m->bd->fPassthrough = aPassthrough; m->bd->fTempEject = aTempEject; m->bd->fNonRotational = aNonRotational; m->bd->fDiscard = aDiscard; m->bd->fImplicit = aImplicit; /* Confirm a successful initialization when it's the case */ autoInitSpan.setSucceeded(); /* Construct a short log name for this attachment. */ Utf8Str ctlName(aControllerName); const char *psz = strpbrk(ctlName.c_str(), " \t:-"); mLogName = Utf8StrFmt("MA%p[%.*s:%u:%u:%s%s]", this, psz ? psz - ctlName.c_str() : 4, ctlName.c_str(), aPort, aDevice, Global::stringifyDeviceType(aType), m->bd->fImplicit ? ":I" : ""); LogFlowThisFunc(("LEAVE - %s\n", getLogName())); return S_OK; } /** * Initializes the medium attachment object given another guest object * (a kind of copy constructor). This object makes a private copy of data * of the original object passed as an argument. */ HRESULT MediumAttachment::initCopy(Machine *aParent, MediumAttachment *aThat) { LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat)); ComAssertRet(aParent && aThat, E_INVALIDARG); /* Enclose the state transition NotReady->InInit->Ready */ AutoInitSpan autoInitSpan(this); AssertReturn(autoInitSpan.isOk(), E_FAIL); m = new Data(aParent); /* m->pPeer is left null */ AutoCaller thatCaller(aThat); AssertComRCReturnRC(thatCaller.rc()); AutoReadLock thatlock(aThat COMMA_LOCKVAL_SRC_POS); m->bd.attachCopy(aThat->m->bd); /* Confirm a successful initialization */ autoInitSpan.setSucceeded(); return S_OK; } /** * Uninitializes the instance. * Called from FinalRelease(). */ void MediumAttachment::uninit() { LogFlowThisFunc(("ENTER - %s\n", getLogName())); /* Enclose the state transition Ready->InUninit->NotReady */ AutoUninitSpan autoUninitSpan(this); if (autoUninitSpan.uninitDone()) return; m->bd.free(); unconst(m->pMachine) = NULL; delete m; m = NULL; LogFlowThisFuncLeave(); } // IHardDiskAttachment properties ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP MediumAttachment::COMGETTER(Medium)(IMedium **aHardDisk) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aHardDisk); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); m->bd->pMedium.queryInterfaceTo(aHardDisk); LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(Controller)(BSTR *aController) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aController); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); /* m->controller is constant during life time, no need to lock */ m->bd->bstrControllerName.cloneTo(aController); LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(Port)(LONG *aPort) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aPort); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); /* m->bd->port is constant during life time, no need to lock */ *aPort = m->bd->lPort; LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(Device)(LONG *aDevice) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aDevice); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); /* m->bd->device is constant during life time, no need to lock */ *aDevice = m->bd->lDevice; LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(Type)(DeviceType_T *aType) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aType); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); /* m->bd->type is constant during life time, no need to lock */ *aType = m->bd->type; LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(Passthrough)(BOOL *aPassthrough) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aPassthrough); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); *aPassthrough = m->bd->fPassthrough; LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(TemporaryEject)(BOOL *aTemporaryEject) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aTemporaryEject); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); *aTemporaryEject = m->bd->fTempEject; LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(IsEjected)(BOOL *aEjected) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aEjected); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); *aEjected = m->fIsEjected; LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(NonRotational)(BOOL *aNonRotational) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aNonRotational); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); *aNonRotational = m->bd->fNonRotational; LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(Discard)(BOOL *aDiscard) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aDiscard); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); *aDiscard = m->bd->fDiscard; LogFlowThisFuncLeave(); return S_OK; } STDMETHODIMP MediumAttachment::COMGETTER(BandwidthGroup) (IBandwidthGroup **aBwGroup) { LogFlowThisFuncEnter(); CheckComArgOutPointerValid(aBwGroup); AutoCaller autoCaller(this); if (FAILED(autoCaller.rc())) return autoCaller.rc(); AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); HRESULT hrc = S_OK; if (m->bd->strBandwidthGroup.isNotEmpty()) { ComObjPtr pBwGroup; hrc = m->pMachine->getBandwidthGroup(m->bd->strBandwidthGroup, pBwGroup, true /* fSetError */); Assert(SUCCEEDED(hrc)); /* This is not allowed to fail because the existence of the group was checked when it was attached. */ if (SUCCEEDED(hrc)) pBwGroup.queryInterfaceTo(aBwGroup); } LogFlowThisFuncLeave(); return hrc; } /** * @note Locks this object for writing. */ void MediumAttachment::rollback() { LogFlowThisFunc(("ENTER - %s\n", getLogName())); /* sanity */ AutoCaller autoCaller(this); AssertComRCReturnVoid(autoCaller.rc()); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); m->bd.rollback(); LogFlowThisFunc(("LEAVE - %s\n", getLogName())); } /** * @note Locks this object for writing. */ void MediumAttachment::commit() { LogFlowThisFunc(("ENTER - %s\n", getLogName())); /* sanity */ AutoCaller autoCaller(this); AssertComRCReturnVoid (autoCaller.rc()); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); if (m->bd.isBackedUp()) m->bd.commit(); LogFlowThisFunc(("LEAVE - %s\n", getLogName())); } bool MediumAttachment::isImplicit() const { return m->bd->fImplicit; } void MediumAttachment::setImplicit(bool aImplicit) { m->bd->fImplicit = aImplicit; } const ComObjPtr& MediumAttachment::getMedium() const { return m->bd->pMedium; } Bstr MediumAttachment::getControllerName() const { return m->bd->bstrControllerName; } LONG MediumAttachment::getPort() const { return m->bd->lPort; } LONG MediumAttachment::getDevice() const { return m->bd->lDevice; } DeviceType_T MediumAttachment::getType() const { return m->bd->type; } bool MediumAttachment::getPassthrough() const { AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); return m->bd->fPassthrough; } bool MediumAttachment::getTempEject() const { AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); return m->bd->fTempEject; } bool MediumAttachment::getNonRotational() const { AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); return m->bd->fNonRotational; } bool MediumAttachment::getDiscard() const { AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS); return m->bd->fDiscard; } const Utf8Str& MediumAttachment::getBandwidthGroup() const { return m->bd->strBandwidthGroup; } bool MediumAttachment::matches(CBSTR aControllerName, LONG aPort, LONG aDevice) { return ( aControllerName == m->bd->bstrControllerName && aPort == m->bd->lPort && aDevice == m->bd->lDevice); } /** * Sets the medium of this attachment and unsets the "implicit" flag. * @param aMedium */ void MediumAttachment::updateMedium(const ComObjPtr &aMedium) { Assert(isWriteLockOnCurrentThread()); m->bd.backup(); m->bd->pMedium = aMedium; m->bd->fImplicit = false; m->fIsEjected = false; } /** Must be called from under this object's write lock. */ void MediumAttachment::updatePassthrough(bool aPassthrough) { Assert(isWriteLockOnCurrentThread()); m->bd.backup(); m->bd->fPassthrough = aPassthrough; } /** Must be called from under this object's write lock. */ void MediumAttachment::updateTempEject(bool aTempEject) { Assert(isWriteLockOnCurrentThread()); m->bd.backup(); m->bd->fTempEject = aTempEject; } /** Must be called from under this object's write lock. */ void MediumAttachment::updateEjected() { Assert(isWriteLockOnCurrentThread()); m->fIsEjected = true; } /** Must be called from under this object's write lock. */ void MediumAttachment::updateNonRotational(bool aNonRotational) { Assert(isWriteLockOnCurrentThread()); m->bd.backup(); m->bd->fNonRotational = aNonRotational; } /** Must be called from under this object's write lock. */ void MediumAttachment::updateDiscard(bool aDiscard) { Assert(isWriteLockOnCurrentThread()); m->bd.backup(); m->bd->fDiscard = aDiscard; } void MediumAttachment::updateBandwidthGroup(const Utf8Str &aBandwidthGroup) { LogFlowThisFuncEnter(); Assert(isWriteLockOnCurrentThread()); m->bd.backup(); m->bd->strBandwidthGroup = aBandwidthGroup; LogFlowThisFuncLeave(); } void MediumAttachment::updateParentMachine(Machine * const pMachine) { LogFlowThisFunc(("ENTER - %s\n", getLogName())); /* sanity */ AutoCaller autoCaller(this); AssertComRCReturnVoid (autoCaller.rc()); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); unconst(m->pMachine) = pMachine; LogFlowThisFunc(("LEAVE - %s\n", getLogName())); }