VirtualBox

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

Last change on this file since 48968 was 48880, checked in by vboxsync, 11 years ago

Main/MediumAttachment: add forgotten method implementation

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