VirtualBox

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

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

Main/MediumAttachment: Update the medium attachment log tag string whenever necessary and add some paranoia checks to catch if someone tries to change attachment settings of a snapshot.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 15.9 KB
Line 
1/* $Id: MediumAttachmentImpl.cpp 61713 2016-06-15 13:15:10Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox COM class implementation
5 */
6
7/*
8 * Copyright (C) 2006-2016 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 i_updateLogName();
148
149 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
150 return S_OK;
151}
152
153/**
154 * Initializes the medium attachment object given another guest object
155 * (a kind of copy constructor). This object makes a private copy of data
156 * of the original object passed as an argument.
157 */
158HRESULT MediumAttachment::initCopy(Machine *aParent, MediumAttachment *aThat)
159{
160 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
161
162 ComAssertRet(aParent && aThat, E_INVALIDARG);
163 Assert(!aParent->i_isSnapshotMachine());
164
165 /* Enclose the state transition NotReady->InInit->Ready */
166 AutoInitSpan autoInitSpan(this);
167 AssertReturn(autoInitSpan.isOk(), E_FAIL);
168
169 m = new Data(aParent);
170 /* m->pPeer is left null */
171
172 AutoCaller thatCaller(aThat);
173 AssertComRCReturnRC(thatCaller.rc());
174
175 AutoReadLock thatlock(aThat COMMA_LOCKVAL_SRC_POS);
176 m->bd.attachCopy(aThat->m->bd);
177
178 /* Confirm a successful initialization */
179 autoInitSpan.setSucceeded();
180
181 /* Construct a short log name for this attachment. */
182 i_updateLogName();
183
184 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
185 return S_OK;
186}
187
188/**
189 * Uninitializes the instance.
190 * Called from FinalRelease().
191 */
192void MediumAttachment::uninit()
193{
194 LogFlowThisFunc(("ENTER - %s\n", i_getLogName()));
195
196 /* Enclose the state transition Ready->InUninit->NotReady */
197 AutoUninitSpan autoUninitSpan(this);
198 if (autoUninitSpan.uninitDone())
199 return;
200
201 m->bd.free();
202
203 unconst(m->pMachine) = NULL;
204
205 delete m;
206 m = NULL;
207
208 LogFlowThisFuncLeave();
209}
210
211// IHardDiskAttachment properties
212/////////////////////////////////////////////////////////////////////////////
213
214
215HRESULT MediumAttachment::getMedium(ComPtr<IMedium> &aHardDisk)
216{
217 LogFlowThisFuncEnter();
218
219 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
220
221 aHardDisk = m->bd->pMedium;
222
223 LogFlowThisFuncLeave();
224 return S_OK;
225}
226
227
228HRESULT MediumAttachment::getController(com::Utf8Str &aController)
229{
230 LogFlowThisFuncEnter();
231
232 /* m->controller is constant during life time, no need to lock */
233 aController = Utf8Str(m->bd->strControllerName);
234
235 LogFlowThisFuncLeave();
236 return S_OK;
237}
238
239
240HRESULT MediumAttachment::getPort(LONG *aPort)
241{
242 LogFlowThisFuncEnter();
243
244 /* m->bd->port is constant during life time, no need to lock */
245 *aPort = m->bd->mData.lPort;
246
247 LogFlowThisFuncLeave();
248 return S_OK;
249}
250
251HRESULT MediumAttachment::getDevice(LONG *aDevice)
252{
253 LogFlowThisFuncEnter();
254
255 /* m->bd->device is constant during life time, no need to lock */
256 *aDevice = m->bd->mData.lDevice;
257
258 LogFlowThisFuncLeave();
259 return S_OK;
260}
261
262HRESULT MediumAttachment::getType(DeviceType_T *aType)
263{
264 LogFlowThisFuncEnter();
265
266 /* m->bd->type is constant during life time, no need to lock */
267 *aType = m->bd->mData.deviceType;
268
269 LogFlowThisFuncLeave();
270 return S_OK;
271}
272
273
274HRESULT MediumAttachment::getPassthrough(BOOL *aPassthrough)
275{
276 LogFlowThisFuncEnter();
277
278 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
279
280 *aPassthrough = m->bd->mData.fPassThrough;
281
282 LogFlowThisFuncLeave();
283 return S_OK;
284}
285
286
287HRESULT MediumAttachment::getTemporaryEject(BOOL *aTemporaryEject)
288{
289 LogFlowThisFuncEnter();
290
291 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
292
293 *aTemporaryEject = m->bd->mData.fTempEject;
294
295 LogFlowThisFuncLeave();
296 return S_OK;
297}
298
299
300HRESULT MediumAttachment::getIsEjected(BOOL *aEjected)
301{
302 LogFlowThisFuncEnter();
303
304 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
305
306 *aEjected = m->fIsEjected;
307
308 LogFlowThisFuncLeave();
309 return S_OK;
310}
311
312
313HRESULT MediumAttachment::getNonRotational(BOOL *aNonRotational)
314{
315 LogFlowThisFuncEnter();
316
317 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
318
319 *aNonRotational = m->bd->mData.fNonRotational;
320
321 LogFlowThisFuncLeave();
322 return S_OK;
323}
324
325HRESULT MediumAttachment::getDiscard(BOOL *aDiscard)
326{
327 LogFlowThisFuncEnter();
328
329 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
330
331 *aDiscard = m->bd->mData.fDiscard;
332
333 LogFlowThisFuncLeave();
334 return S_OK;
335}
336
337
338HRESULT MediumAttachment::getBandwidthGroup(ComPtr<IBandwidthGroup> &aBandwidthGroup)
339{
340 LogFlowThisFuncEnter();
341
342 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
343
344 HRESULT hrc = S_OK;
345 if (m->bd->mData.strBwGroup.isNotEmpty())
346 {
347 ComObjPtr<BandwidthGroup> pBwGroup;
348 hrc = m->pMachine->i_getBandwidthGroup(m->bd->mData.strBwGroup, pBwGroup, true /* fSetError */);
349
350 Assert(SUCCEEDED(hrc)); /* This is not allowed to fail because the existence of the
351 group was checked when it was attached. */
352
353 if (SUCCEEDED(hrc))
354 pBwGroup.queryInterfaceTo(aBandwidthGroup.asOutParam());
355 }
356
357 LogFlowThisFuncLeave();
358 return hrc;
359}
360
361HRESULT MediumAttachment::getHotPluggable(BOOL *aHotPluggable)
362{
363 LogFlowThisFuncEnter();
364
365 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
366
367 *aHotPluggable = m->bd->mData.fHotPluggable;
368
369 LogFlowThisFuncLeave();
370 return S_OK;
371}
372
373/**
374 * @note Locks this object for writing.
375 */
376void MediumAttachment::i_rollback()
377{
378 LogFlowThisFunc(("ENTER - %s\n", i_getLogName()));
379
380 /* sanity */
381 AutoCaller autoCaller(this);
382 AssertComRCReturnVoid(autoCaller.rc());
383
384 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
385
386 m->bd.rollback();
387
388 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
389}
390
391/**
392 * @note Locks this object for writing.
393 */
394void MediumAttachment::i_commit()
395{
396 LogFlowThisFunc(("ENTER - %s\n", i_getLogName()));
397
398 /* sanity */
399 AutoCaller autoCaller(this);
400 AssertComRCReturnVoid(autoCaller.rc());
401
402 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
403
404 if (m->bd.isBackedUp())
405 m->bd.commit();
406
407 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
408}
409
410bool MediumAttachment::i_isImplicit() const
411{
412 return m->bd->fImplicit;
413}
414
415void MediumAttachment::i_setImplicit(bool aImplicit)
416{
417 Assert(!m->pMachine->i_isSnapshotMachine());
418 m->bd->fImplicit = aImplicit;
419
420 /* Construct a short log name for this attachment. */
421 i_updateLogName();
422}
423
424const ComObjPtr<Medium>& MediumAttachment::i_getMedium() const
425{
426 return m->bd->pMedium;
427}
428
429const Utf8Str &MediumAttachment::i_getControllerName() const
430{
431 return m->bd->strControllerName;
432}
433
434LONG MediumAttachment::i_getPort() const
435{
436 return m->bd->mData.lPort;
437}
438
439LONG MediumAttachment::i_getDevice() const
440{
441 return m->bd->mData.lDevice;
442}
443
444DeviceType_T MediumAttachment::i_getType() const
445{
446 return m->bd->mData.deviceType;
447}
448
449bool MediumAttachment::i_getPassthrough() const
450{
451 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
452 return m->bd->mData.fPassThrough;
453}
454
455bool MediumAttachment::i_getTempEject() const
456{
457 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
458 return m->bd->mData.fTempEject;
459}
460
461bool MediumAttachment::i_getNonRotational() const
462{
463 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
464 return m->bd->mData.fNonRotational;
465}
466
467bool MediumAttachment::i_getDiscard() const
468{
469 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
470 return m->bd->mData.fDiscard;
471}
472
473bool MediumAttachment::i_getHotPluggable() const
474{
475 AutoReadLock lock(this COMMA_LOCKVAL_SRC_POS);
476 return m->bd->mData.fHotPluggable;
477}
478
479Utf8Str& MediumAttachment::i_getBandwidthGroup() const
480{
481 return m->bd->mData.strBwGroup;
482}
483
484bool MediumAttachment::i_matches(const Utf8Str &aControllerName, LONG aPort, LONG aDevice)
485{
486 return ( aControllerName == m->bd->strControllerName
487 && aPort == m->bd->mData.lPort
488 && aDevice == m->bd->mData.lDevice);
489}
490
491/** Must be called from under this object's write lock. */
492void MediumAttachment::i_updateName(const Utf8Str &aName)
493{
494 Assert(isWriteLockOnCurrentThread());
495 Assert(!m->pMachine->i_isSnapshotMachine());
496
497 m->bd.backup();
498 unconst(m->bd->strControllerName) = aName;
499
500 /* Construct a short log name for this attachment. */
501 i_updateLogName();
502}
503
504/**
505 * Sets the medium of this attachment and unsets the "implicit" flag.
506 * @param aMedium
507 */
508void MediumAttachment::i_updateMedium(const ComObjPtr<Medium> &aMedium)
509{
510 Assert(isWriteLockOnCurrentThread());
511 Assert(!m->pMachine->i_isSnapshotMachine());
512
513 m->bd.backup();
514 m->bd->pMedium = aMedium;
515 m->bd->fImplicit = false;
516 m->fIsEjected = false;
517}
518
519/** Must be called from under this object's write lock. */
520void MediumAttachment::i_updatePassthrough(bool aPassthrough)
521{
522 Assert(isWriteLockOnCurrentThread());
523 Assert(!m->pMachine->i_isSnapshotMachine());
524
525 m->bd.backup();
526 m->bd->mData.fPassThrough = aPassthrough;
527}
528
529/** Must be called from under this object's write lock. */
530void MediumAttachment::i_updateTempEject(bool aTempEject)
531{
532 Assert(isWriteLockOnCurrentThread());
533 Assert(!m->pMachine->i_isSnapshotMachine());
534
535 m->bd.backup();
536 m->bd->mData.fTempEject = aTempEject;
537}
538
539/** Must be called from under this object's write lock. */
540void MediumAttachment::i_updateEjected()
541{
542 Assert(isWriteLockOnCurrentThread());
543 Assert(!m->pMachine->i_isSnapshotMachine());
544
545 m->fIsEjected = true;
546}
547
548/** Must be called from under this object's write lock. */
549void MediumAttachment::i_updateNonRotational(bool aNonRotational)
550{
551 Assert(isWriteLockOnCurrentThread());
552 Assert(!m->pMachine->i_isSnapshotMachine());
553
554 m->bd.backup();
555 m->bd->mData.fNonRotational = aNonRotational;
556}
557
558/** Must be called from under this object's write lock. */
559void MediumAttachment::i_updateDiscard(bool aDiscard)
560{
561 Assert(isWriteLockOnCurrentThread());
562 Assert(!m->pMachine->i_isSnapshotMachine());
563
564 m->bd.backup();
565 m->bd->mData.fDiscard = aDiscard;
566}
567
568/** Must be called from under this object's write lock. */
569void MediumAttachment::i_updateHotPluggable(bool aHotPluggable)
570{
571 Assert(isWriteLockOnCurrentThread());
572 Assert(!m->pMachine->i_isSnapshotMachine());
573
574 m->bd.backup();
575 m->bd->mData.fHotPluggable = aHotPluggable;
576}
577
578void MediumAttachment::i_updateBandwidthGroup(const Utf8Str &aBandwidthGroup)
579{
580 LogFlowThisFuncEnter();
581 Assert(isWriteLockOnCurrentThread());
582 Assert(!m->pMachine->i_isSnapshotMachine());
583
584 m->bd.backup();
585 m->bd->mData.strBwGroup = aBandwidthGroup;
586
587 LogFlowThisFuncLeave();
588}
589
590void MediumAttachment::i_updateParentMachine(Machine * const pMachine)
591{
592 LogFlowThisFunc(("ENTER - %s\n", i_getLogName()));
593 /* sanity */
594 AutoCaller autoCaller(this);
595 AssertComRCReturnVoid(autoCaller.rc());
596 Assert(!m->pMachine->i_isSnapshotMachine());
597
598 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
599
600 unconst(m->pMachine) = pMachine;
601
602 LogFlowThisFunc(("LEAVE - %s\n", i_getLogName()));
603}
604
605void MediumAttachment::i_updateLogName()
606{
607 const char *pszName = m->bd->strControllerName.c_str();
608 const char *pszEndNick = strpbrk(pszName, " \t:-");
609 mLogName = Utf8StrFmt("MA%p[%.*s:%u:%u:%s%s]",
610 this,
611 pszEndNick ? pszEndNick - pszName : 4, pszName,
612 m->bd->mData.lPort, m->bd->mData.lDevice, Global::stringifyDeviceType(m->bd->mData.deviceType),
613 m->bd->fImplicit ? ":I" : "");
614}
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