/* $Id: ParallelPortImpl.cpp 54971 2015-03-26 16:40:30Z vboxsync $ */ /** @file * VirtualBox COM class implementation */ /* * Copyright (C) 2006-2015 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 "ParallelPortImpl.h" #include "MachineImpl.h" #include "VirtualBoxImpl.h" #include #include #include #include "AutoStateDep.h" #include "AutoCaller.h" #include "Logging.h" //////////////////////////////////////////////////////////////////////////////// // // ParallelPort private data definition // //////////////////////////////////////////////////////////////////////////////// struct ParallelPort::Data { Data() : fModified(false), pMachine(NULL) { } bool fModified; Machine * const pMachine; const ComObjPtr pPeer; Backupable bd; }; // constructor / destructor ///////////////////////////////////////////////////////////////////////////// DEFINE_EMPTY_CTOR_DTOR(ParallelPort) HRESULT ParallelPort::FinalConstruct() { return BaseFinalConstruct(); } void ParallelPort::FinalRelease() { uninit(); BaseFinalRelease(); } // public initializer/uninitializer for internal purposes only ///////////////////////////////////////////////////////////////////////////// /** * Initializes the Parallel Port object. * * @param aParent Handle of the parent object. */ HRESULT ParallelPort::init(Machine *aParent, ULONG aSlot) { LogFlowThisFunc(("aParent=%p, aSlot=%d\n", aParent, aSlot)); ComAssertRet(aParent, 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->pPeer is left null */ m->bd.allocate(); /* initialize data */ m->bd->ulSlot = aSlot; /* Confirm a successful initialization */ autoInitSpan.setSucceeded(); return S_OK; } /** * Initializes the Parallel Port object given another serial port object * (a kind of copy constructor). This object shares data with * the object passed as an argument. * * @note This object must be destroyed before the original object * it shares data with is destroyed. * * @note Locks @a aThat object for reading. */ HRESULT ParallelPort::init(Machine *aParent, ParallelPort *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; unconst(m->pMachine) = aParent; unconst(m->pPeer) = aThat; AutoCaller thatCaller (aThat); AssertComRCReturnRC(thatCaller.rc()); AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS); m->bd.share(aThat->m->bd); /* Confirm a successful initialization */ autoInitSpan.setSucceeded(); return S_OK; } /** * Initializes the guest 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. * * @note Locks @a aThat object for reading. */ HRESULT ParallelPort::initCopy(Machine *aParent, ParallelPort *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; unconst(m->pMachine) = 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 and sets the ready flag to FALSE. * Called either from FinalRelease() or by the parent when it gets destroyed. */ void ParallelPort::uninit() { LogFlowThisFunc(("\n")); /* Enclose the state transition Ready->InUninit->NotReady */ AutoUninitSpan autoUninitSpan(this); if (autoUninitSpan.uninitDone()) return; m->bd.free(); unconst(m->pPeer) = NULL; unconst(m->pMachine) = NULL; delete m; m = NULL; } // IParallelPort properties ///////////////////////////////////////////////////////////////////////////// HRESULT ParallelPort::getEnabled(BOOL *aEnabled) { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); *aEnabled = m->bd->fEnabled; return S_OK; } HRESULT ParallelPort::setEnabled(BOOL aEnabled) { LogFlowThisFunc(("aEnabled=%RTbool\n", aEnabled)); /* the machine needs to be mutable */ AutoMutableStateDependency adep(m->pMachine); if (FAILED(adep.rc())) return adep.rc(); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); if (m->bd->fEnabled != !!aEnabled) { if (aEnabled && m->bd->strPath.isEmpty()) return setError(E_INVALIDARG, tr("Cannot enable the parallel port %d because the port path is empty or null"), m->bd->ulSlot); m->bd.backup(); m->bd->fEnabled = !!aEnabled; m->fModified = true; // leave the lock before informing callbacks alock.release(); AutoWriteLock mlock(m->pMachine COMMA_LOCKVAL_SRC_POS); m->pMachine->i_setModified(Machine::IsModified_ParallelPorts); mlock.release(); m->pMachine->i_onParallelPortChange(this); } return S_OK; } HRESULT ParallelPort::getSlot(ULONG *aSlot) { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); *aSlot = m->bd->ulSlot; return S_OK; } HRESULT ParallelPort::getIRQ(ULONG *aIRQ) { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); *aIRQ = m->bd->ulIRQ; return S_OK; } HRESULT ParallelPort::setIRQ(ULONG aIRQ) { /* check IRQ limits * (when changing this, make sure it corresponds to XML schema */ if (aIRQ > 255) return setError(E_INVALIDARG, tr("Invalid IRQ number of the parallel port %d: %lu (must be in range [0, %lu])"), m->bd->ulSlot, aIRQ, 255); /* the machine needs to be mutable */ AutoMutableStateDependency adep(m->pMachine); if (FAILED(adep.rc())) return adep.rc(); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); if (m->bd->ulIRQ != aIRQ) { m->bd.backup(); m->bd->ulIRQ = aIRQ; m->fModified = true; // leave the lock before informing callbacks alock.release(); AutoWriteLock mlock(m->pMachine COMMA_LOCKVAL_SRC_POS); m->pMachine->i_setModified(Machine::IsModified_ParallelPorts); mlock.release(); m->pMachine->i_onParallelPortChange(this); } return S_OK; } HRESULT ParallelPort::getIOBase(ULONG *aIOBase) { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); *aIOBase = m->bd->ulIOBase; return S_OK; } HRESULT ParallelPort::setIOBase(ULONG aIOBase) { /* check IOBase limits * (when changing this, make sure it corresponds to XML schema */ if (aIOBase > 0xFFFF) return setError(E_INVALIDARG, tr("Invalid I/O port base address of the parallel port %d: %lu (must be in range [0, 0x%X])"), m->bd->ulSlot, aIOBase, 0, 0xFFFF); /* the machine needs to be mutable */ AutoMutableStateDependency adep(m->pMachine); if (FAILED(adep.rc())) return adep.rc(); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); if (m->bd->ulIOBase != aIOBase) { m->bd.backup(); m->bd->ulIOBase = aIOBase; m->fModified = true; // leave the lock before informing callbacks alock.release(); AutoWriteLock mlock(m->pMachine COMMA_LOCKVAL_SRC_POS); m->pMachine->i_setModified(Machine::IsModified_ParallelPorts); mlock.release(); m->pMachine->i_onParallelPortChange(this); } return S_OK; } HRESULT ParallelPort::getPath(com::Utf8Str &aPath) { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); aPath = m->bd->strPath; return S_OK; } HRESULT ParallelPort::setPath(const com::Utf8Str &aPath) { /* the machine needs to be mutable */ AutoMutableOrSavedStateDependency adep(m->pMachine); if (FAILED(adep.rc())) return adep.rc(); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); if (aPath != m->bd->strPath) { HRESULT rc = i_checkSetPath(aPath); if (FAILED(rc)) return rc; m->bd.backup(); m->bd->strPath = aPath; m->fModified = true; // leave the lock before informing callbacks alock.release(); AutoWriteLock mlock(m->pMachine COMMA_LOCKVAL_SRC_POS); m->pMachine->i_setModified(Machine::IsModified_ParallelPorts); mlock.release(); return m->pMachine->i_onParallelPortChange(this); } return S_OK; } // public methods only for internal purposes //////////////////////////////////////////////////////////////////////////////// /** * Loads settings from the given port node. * May be called once right after this object creation. * * @param aPortNode node. * * @note Locks this object for writing. */ HRESULT ParallelPort::i_loadSettings(const settings::ParallelPort &data) { AutoCaller autoCaller(this); AssertComRCReturnRC(autoCaller.rc()); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); // simply copy *m->bd.data() = data; return S_OK; } /** * Saves settings to the given port node. * * Note that the given Port node is completely empty on input. * * @param node. * * @note Locks this object for reading. */ HRESULT ParallelPort::i_saveSettings(settings::ParallelPort &data) { AutoCaller autoCaller(this); AssertComRCReturnRC(autoCaller.rc()); AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); // simply copy data = *m->bd.data(); return S_OK; } /** * Returns true if any setter method has modified settings of this instance. * @return */ bool ParallelPort::i_isModified() { AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); return m->fModified; } /** * @note Locks this object for writing. */ void ParallelPort::i_rollback() { /* sanity */ AutoCaller autoCaller(this); AssertComRCReturnVoid(autoCaller.rc()); AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); m->bd.rollback(); } /** * @note Locks this object for writing, together with the peer object (also * for writing) if there is one. */ void ParallelPort::i_commit() { /* sanity */ AutoCaller autoCaller(this); AssertComRCReturnVoid (autoCaller.rc()); /* sanity too */ AutoCaller peerCaller (m->pPeer); AssertComRCReturnVoid (peerCaller.rc()); /* lock both for writing since we modify both (m->pPeer is "master" so locked * first) */ AutoMultiWriteLock2 alock(m->pPeer, this COMMA_LOCKVAL_SRC_POS); if (m->bd.isBackedUp()) { m->bd.commit(); if (m->pPeer) { /* attach new data to the peer and reshare it */ m->pPeer->m->bd.attach(m->bd); } } } /** * @note Locks this object for writing, together with the peer object * represented by @a aThat (locked for reading). */ void ParallelPort::i_copyFrom(ParallelPort *aThat) { AssertReturnVoid (aThat != NULL); /* sanity */ AutoCaller autoCaller(this); AssertComRCReturnVoid (autoCaller.rc()); /* sanity too */ AutoCaller thatCaller (aThat); AssertComRCReturnVoid (thatCaller.rc()); /* peer is not modified, lock it for reading (aThat is "master" so locked * first) */ AutoReadLock rl(aThat COMMA_LOCKVAL_SRC_POS); AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS); /* this will back up current data */ m->bd.assignCopy(aThat->m->bd); } /** * Validates COMSETTER(Path) arguments. */ HRESULT ParallelPort::i_checkSetPath(const Utf8Str &str) { AssertReturn(isWriteLockOnCurrentThread(), E_FAIL); if ( m->bd->fEnabled && str.isEmpty() ) return setError(E_INVALIDARG, tr("Path of the parallel port %d may not be empty or null " "when the port is enabled"), m->bd->ulSlot); return S_OK; } /* vi: set tabstop=4 shiftwidth=4 expandtab: */