VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/MediumAttachmentImpl.cpp@ 61377

Last change on this file since 61377 was 61377, checked in by vboxsync, 9 years ago

bugref:8238 VBoxSVC settings - MediumAttachementImpl.cpp

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 15.0 KB
Line 
1/* $Id: MediumAttachmentImpl.cpp 61377 2016-06-01 14:20:24Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox COM class implementation
5 */
6
7/*
8 * Copyright (C) 2006-2015 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#include "MediumAttachmentImpl.h"
20#include "MachineImpl.h"
21#include "MediumImpl.h"
22#include "Global.h"
23
24#include "AutoCaller.h"
25#include "Logging.h"
26
27#include <iprt/cpp/utils.h>
28
29////////////////////////////////////////////////////////////////////////////////
30//
31// private member data definition
32//
33////////////////////////////////////////////////////////////////////////////////
34
35struct BackupableMediumAttachmentData
36{
37 BackupableMediumAttachmentData()
38 : fImplicit(false)
39 { }
40
41 ComObjPtr<Medium> pMedium;
42 /* Since MediumAttachment is not a first class citizen when it
43 * comes to managing settings, having a reference to the storage
44 * controller will not work - when settings are changed it will point
45 * to the old, uninitialized instance. Changing this requires
46 * substantial changes to MediumImpl.cpp. */
47 /* Same counts for the assigned bandwidth group */
48 bool fImplicit;
49 const Utf8Str strControllerName;
50 settings::AttachedDevice mData;
51};
52
53struct MediumAttachment::Data
54{
55 Data(Machine * const aMachine = NULL)
56 : pMachine(aMachine),
57 fIsEjected(false)
58 { }
59
60 /** Reference to Machine object, for checking mutable state. */
61 Machine * const pMachine;
62 /* later: const ComObjPtr<MediumAttachment> mPeer; */
63 bool fIsEjected;
64 Backupable<BackupableMediumAttachmentData> bd;
65};
66
67// constructor / destructor
68/////////////////////////////////////////////////////////////////////////////
69
70DEFINE_EMPTY_CTOR_DTOR(MediumAttachment)
71
72HRESULT MediumAttachment::FinalConstruct()
73{
74 LogFlowThisFunc(("\n"));
75 return BaseFinalConstruct();
76}
77
78void MediumAttachment::FinalRelease()
79{
80 LogFlowThisFuncEnter();
81 uninit();
82 BaseFinalRelease();
83 LogFlowThisFuncLeave();
84}
85
86// public initializer/uninitializer for internal purposes only
87/////////////////////////////////////////////////////////////////////////////
88
89/**
90 * Initializes the medium attachment object.
91 *
92 * @param aParent Machine object.
93 * @param aMedium Medium object.
94 * @param aController Controller the hard disk is attached to.
95 * @param aPort Port number.
96 * @param aDevice Device number on the port.
97 * @param aPassthrough Whether accesses are directly passed to the host drive.
98 * @param aBandwidthLimit Bandwidth limit in Mbps
99 */
100HRESULT MediumAttachment::init(Machine *aParent,
101 Medium *aMedium,
102 const Utf8Str &aControllerName,
103 LONG aPort,
104 LONG aDevice,
105 DeviceType_T aType,
106 bool aImplicit,
107 bool aPassthrough,
108 bool aTempEject,
109 bool aNonRotational,
110 bool aDiscard,
111 bool aHotPluggable,
112 const Utf8Str &strBandwidthGroup)
113{
114 LogFlowThisFuncEnter();
115 LogFlowThisFunc(("aParent=%p aMedium=%p aControllerName=%s aPort=%d aDevice=%d aType=%d aImplicit=%d aPassthrough=%d aTempEject=%d aNonRotational=%d aDiscard=%d aHotPluggable=%d strBandwithGroup=%s\n", aParent, aMedium, aControllerName.c_str(), aPort, aDevice, aType, aImplicit, aPassthrough, aTempEject, aNonRotational, aDiscard, aHotPluggable, strBandwidthGroup.c_str()));
116
117 if (aType == DeviceType_HardDisk)
118 AssertReturn(aMedium, E_INVALIDARG);
119
120 /* Enclose the state transition NotReady->InInit->Ready */
121 AutoInitSpan autoInitSpan(this);
122 AssertReturn(autoInitSpan.isOk(), E_FAIL);
123
124 m = new Data();
125
126 unconst(m->pMachine) = aParent;
127
128 m->bd.allocate();
129 m->bd->pMedium = aMedium;
130 m->bd->mData.strBwGroup = strBandwidthGroup;
131 unconst(m->bd->strControllerName) = aControllerName;
132 m->bd->mData.lPort = aPort;
133 m->bd->mData.lDevice = aDevice;
134 m->bd->mData.deviceType = aType;
135
136 m->bd->mData.fPassThrough = aPassthrough;
137 m->bd->mData.fTempEject = aTempEject;
138 m->bd->mData.fNonRotational = aNonRotational;
139 m->bd->mData.fDiscard = aDiscard;
140 m->bd->fImplicit = aImplicit;
141 m->bd->mData.fHotPluggable = aHotPluggable;
142
143 /* Confirm a successful initialization when it's the case */
144 autoInitSpan.setSucceeded();
145
146 /* Construct a short log name for this attachment. */
147 Utf8Str ctlName(aControllerName);
148 const char *psz = strpbrk(ctlName.c_str(), " \t:-");
149 mLogName = Utf8StrFmt("MA%p[%.*s:%u:%u:%s%s]",
150 this,
151 psz ? psz - ctlName.c_str() : 4, ctlName.c_str(),
152 aPort, aDevice, Global::stringifyDeviceType(aType),
153 m->bd->fImplicit ? ":I" : "");
154
155 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
156 return S_OK;
157}
158
159/**
160 * Initializes the medium attachment object given another guest object
161 * (a kind of copy constructor). This object makes a private copy of data
162 * of the original object passed as an argument.
163 */
164HRESULT MediumAttachment::initCopy(Machine *aParent, MediumAttachment *aThat)
165{
166 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
167
168 ComAssertRet(aParent && aThat, E_INVALIDARG);
169
170 /* Enclose the state transition NotReady->InInit->Ready */
171 AutoInitSpan autoInitSpan(this);
172 AssertReturn(autoInitSpan.isOk(), E_FAIL);
173
174 m = new Data(aParent);
175 /* m->pPeer is left null */
176
177 AutoCaller thatCaller(aThat);
178 AssertComRCReturnRC(thatCaller.rc());
179
180 AutoReadLock thatlock(aThat COMMA_LOCKVAL_SRC_POS);
181 m->bd.attachCopy(aThat->m->bd);
182
183 /* Confirm a successful initialization */
184 autoInitSpan.setSucceeded();
185
186 return S_OK;
187}
188
189/**
190 * Uninitializes the instance.
191 * Called from FinalRelease().
192 */
193void MediumAttachment::uninit()
194{
195 LogFlowThisFunc(("ENTER - %s\n", i_getLogName()));
196
197 /* Enclose the state transition Ready->InUninit->NotReady */
198 AutoUninitSpan autoUninitSpan(this);
199 if (autoUninitSpan.uninitDone())
200 return;
201
202 m->bd.free();
203
204 unconst(m->pMachine) = NULL;
205
206 delete m;
207 m = NULL;
208
209 LogFlowThisFuncLeave();
210}
211
212// IHardDiskAttachment properties
213/////////////////////////////////////////////////////////////////////////////
214
215
216HRESULT MediumAttachment::getMedium(ComPtr<IMedium> &aHardDisk)
217{
218 LogFlowThisFuncEnter();
219
220 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
221
222 aHardDisk = m->bd->pMedium;
223
224 LogFlowThisFuncLeave();
225 return S_OK;
226}
227
228
229HRESULT MediumAttachment::getController(com::Utf8Str &aController)
230{
231 LogFlowThisFuncEnter();
232
233 /* m->controller is constant during life time, no need to lock */
234 aController = Utf8Str(m->bd->strControllerName);
235
236 LogFlowThisFuncLeave();
237 return S_OK;
238}
239
240
241HRESULT MediumAttachment::getPort(LONG *aPort)
242{
243 LogFlowThisFuncEnter();
244
245 /* m->bd->port is constant during life time, no need to lock */
246 *aPort = m->bd->mData.lPort;
247
248 LogFlowThisFuncLeave();
249 return S_OK;
250}
251
252HRESULT MediumAttachment::getDevice(LONG *aDevice)
253{
254 LogFlowThisFuncEnter();
255
256 /* m->bd->device is constant during life time, no need to lock */
257 *aDevice = m->bd->mData.lDevice;
258
259 LogFlowThisFuncLeave();
260 return S_OK;
261}
262
263HRESULT MediumAttachment::getType(DeviceType_T *aType)
264{
265 LogFlowThisFuncEnter();
266
267 /* m->bd->type is constant during life time, no need to lock */
268 *aType = m->bd->mData.deviceType;
269
270 LogFlowThisFuncLeave();
271 return S_OK;
272}
273
274
275HRESULT MediumAttachment::getPassthrough(BOOL *aPassthrough)
276{
277 LogFlowThisFuncEnter();
278
279 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
280
281 *aPassthrough = m->bd->mData.fPassThrough;
282
283 LogFlowThisFuncLeave();
284 return S_OK;
285}
286
287
288HRESULT MediumAttachment::getTemporaryEject(BOOL *aTemporaryEject)
289{
290 LogFlowThisFuncEnter();
291
292 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
293
294 *aTemporaryEject = m->bd->mData.fTempEject;
295
296 LogFlowThisFuncLeave();
297 return S_OK;
298}
299
300
301HRESULT MediumAttachment::getIsEjected(BOOL *aEjected)
302{
303 LogFlowThisFuncEnter();
304
305 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
306
307 *aEjected = m->fIsEjected;
308
309 LogFlowThisFuncLeave();
310 return S_OK;
311}
312
313
314HRESULT MediumAttachment::getNonRotational(BOOL *aNonRotational)
315{
316 LogFlowThisFuncEnter();
317
318 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
319
320 *aNonRotational = m->bd->mData.fNonRotational;
321
322 LogFlowThisFuncLeave();
323 return S_OK;
324}
325
326HRESULT MediumAttachment::getDiscard(BOOL *aDiscard)
327{
328 LogFlowThisFuncEnter();
329
330 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
331
332 *aDiscard = m->bd->mData.fDiscard;
333
334 LogFlowThisFuncLeave();
335 return S_OK;
336}
337
338
339HRESULT MediumAttachment::getBandwidthGroup(ComPtr<IBandwidthGroup> &aBandwidthGroup)
340{
341 LogFlowThisFuncEnter();
342
343 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
344
345 HRESULT hrc = S_OK;
346 if (m->bd->mData.strBwGroup.isNotEmpty())
347 {
348 ComObjPtr<BandwidthGroup> pBwGroup;
349 hrc = m->pMachine->i_getBandwidthGroup(m->bd->mData.strBwGroup, pBwGroup, true /* fSetError */);
350
351 Assert(SUCCEEDED(hrc)); /* This is not allowed to fail because the existence of the
352 group was checked when it was attached. */
353
354 if (SUCCEEDED(hrc))
355 pBwGroup.queryInterfaceTo(aBandwidthGroup.asOutParam());
356 }
357
358 LogFlowThisFuncLeave();
359 return hrc;
360}
361
362HRESULT MediumAttachment::getHotPluggable(BOOL *aHotPluggable)
363{
364 LogFlowThisFuncEnter();
365
366 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
367
368 *aHotPluggable = m->bd->mData.fHotPluggable;
369
370 LogFlowThisFuncLeave();
371 return S_OK;
372}
373
374/**
375 * @note Locks this object for writing.
376 */
377void MediumAttachment::i_rollback()
378{
379 LogFlowThisFunc(("ENTER - %s\n", i_getLogName()));
380
381 /* sanity */
382 AutoCaller autoCaller(this);
383 AssertComRCReturnVoid(autoCaller.rc());
384
385 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
386
387 m->bd.rollback();
388
389 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
390}
391
392/**
393 * @note Locks this object for writing.
394 */
395void MediumAttachment::i_commit()
396{
397 LogFlowThisFunc(("ENTER - %s\n", i_getLogName()));
398
399 /* sanity */
400 AutoCaller autoCaller(this);
401 AssertComRCReturnVoid (autoCaller.rc());
402
403 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
404
405 if (m->bd.isBackedUp())
406 m->bd.commit();
407
408 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
409}
410
411bool MediumAttachment::i_isImplicit() const
412{
413 return m->bd->fImplicit;
414}
415
416void MediumAttachment::i_setImplicit(bool aImplicit)
417{
418 m->bd->fImplicit = aImplicit;
419}
420
421const ComObjPtr<Medium>& MediumAttachment::i_getMedium() const
422{
423 return m->bd->pMedium;
424}
425
426const Utf8Str &MediumAttachment::i_getControllerName() const
427{
428 return m->bd->strControllerName;
429}
430
431LONG MediumAttachment::i_getPort() const
432{
433 return m->bd->mData.lPort;
434}
435
436LONG MediumAttachment::i_getDevice() const
437{
438 return m->bd->mData.lDevice;
439}
440
441DeviceType_T MediumAttachment::i_getType() const
442{
443 return m->bd->mData.deviceType;
444}
445
446bool MediumAttachment::i_getPassthrough() const
447{
448 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
449 return m->bd->mData.fPassThrough;
450}
451
452bool MediumAttachment::i_getTempEject() const
453{
454 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
455 return m->bd->mData.fTempEject;
456}
457
458bool MediumAttachment::i_getNonRotational() const
459{
460 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
461 return m->bd->mData.fNonRotational;
462}
463
464bool MediumAttachment::i_getDiscard() const
465{
466 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
467 return m->bd->mData.fDiscard;
468}
469
470bool MediumAttachment::i_getHotPluggable() const
471{
472 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
473 return m->bd->mData.fHotPluggable;
474}
475
476Utf8Str& MediumAttachment::i_getBandwidthGroup() const
477{
478 return m->bd->mData.strBwGroup;
479}
480
481bool MediumAttachment::i_matches(const Utf8Str &aControllerName, LONG aPort, LONG aDevice)
482{
483 return ( aControllerName == m->bd->strControllerName
484 && aPort == m->bd->mData.lPort
485 && aDevice == m->bd->mData.lDevice);
486}
487
488/** Must be called from under this object's write lock. */
489void MediumAttachment::i_updateName(const Utf8Str &aName)
490{
491 Assert(isWriteLockOnCurrentThread());
492
493 m->bd.backup();
494 unconst(m->bd->strControllerName) = aName;
495}
496
497/**
498 * Sets the medium of this attachment and unsets the "implicit" flag.
499 * @param aMedium
500 */
501void MediumAttachment::i_updateMedium(const ComObjPtr<Medium> &aMedium)
502{
503 Assert(isWriteLockOnCurrentThread());
504
505 m->bd.backup();
506 m->bd->pMedium = aMedium;
507 m->bd->fImplicit = false;
508 m->fIsEjected = false;
509}
510
511/** Must be called from under this object's write lock. */
512void MediumAttachment::i_updatePassthrough(bool aPassthrough)
513{
514 Assert(isWriteLockOnCurrentThread());
515
516 m->bd.backup();
517 m->bd->mData.fPassThrough = aPassthrough;
518}
519
520/** Must be called from under this object's write lock. */
521void MediumAttachment::i_updateTempEject(bool aTempEject)
522{
523 Assert(isWriteLockOnCurrentThread());
524
525 m->bd.backup();
526 m->bd->mData.fTempEject = aTempEject;
527}
528
529/** Must be called from under this object's write lock. */
530void MediumAttachment::i_updateEjected()
531{
532 Assert(isWriteLockOnCurrentThread());
533
534 m->fIsEjected = true;
535}
536
537/** Must be called from under this object's write lock. */
538void MediumAttachment::i_updateNonRotational(bool aNonRotational)
539{
540 Assert(isWriteLockOnCurrentThread());
541
542 m->bd.backup();
543 m->bd->mData.fNonRotational = aNonRotational;
544}
545
546/** Must be called from under this object's write lock. */
547void MediumAttachment::i_updateDiscard(bool aDiscard)
548{
549 Assert(isWriteLockOnCurrentThread());
550
551 m->bd.backup();
552 m->bd->mData.fDiscard = aDiscard;
553}
554
555/** Must be called from under this object's write lock. */
556void MediumAttachment::i_updateHotPluggable(bool aHotPluggable)
557{
558 Assert(isWriteLockOnCurrentThread());
559
560 m->bd.backup();
561 m->bd->mData.fHotPluggable = aHotPluggable;
562}
563
564void MediumAttachment::i_updateBandwidthGroup(const Utf8Str &aBandwidthGroup)
565{
566 LogFlowThisFuncEnter();
567 Assert(isWriteLockOnCurrentThread());
568
569 m->bd.backup();
570 m->bd->mData.strBwGroup = aBandwidthGroup;
571
572 LogFlowThisFuncLeave();
573}
574
575void MediumAttachment::i_updateParentMachine(Machine * const pMachine)
576{
577 LogFlowThisFunc(("ENTER - %s\n", i_getLogName()));
578 /* sanity */
579 AutoCaller autoCaller(this);
580 AssertComRCReturnVoid (autoCaller.rc());
581
582 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
583
584 unconst(m->pMachine) = pMachine;
585
586 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
587}
588
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette