VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/StorageControllerImpl.cpp@ 64972

Last change on this file since 64972 was 63975, checked in by vboxsync, 8 years ago

Main: fixes initialization of SAS controller (ticketref:15972)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 22.8 KB
Line 
1/* $Id: StorageControllerImpl.cpp 63975 2016-09-23 13:28:36Z vboxsync $ */
2
3/** @file
4 *
5 * Implementation of IStorageController.
6 */
7
8/*
9 * Copyright (C) 2008-2016 Oracle Corporation
10 *
11 * This file is part of VirtualBox Open Source Edition (OSE), as
12 * available from http://www.virtualbox.org. This file is free software;
13 * you can redistribute it and/or modify it under the terms of the GNU
14 * General Public License (GPL) as published by the Free Software
15 * Foundation, in version 2 as it comes in the "COPYING" file of the
16 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18 */
19
20#include "StorageControllerImpl.h"
21#include "MachineImpl.h"
22#include "VirtualBoxImpl.h"
23#include "SystemPropertiesImpl.h"
24
25#include <iprt/string.h>
26#include <iprt/cpp/utils.h>
27
28#include <VBox/err.h>
29#include <VBox/settings.h>
30
31#include <algorithm>
32
33#include "AutoStateDep.h"
34#include "AutoCaller.h"
35#include "Logging.h"
36
37// defines
38/////////////////////////////////////////////////////////////////////////////
39
40struct StorageController::Data
41{
42 Data(Machine * const aMachine)
43 : pVirtualBox(NULL),
44 pSystemProperties(NULL),
45 pParent(aMachine)
46 {
47 unconst(pVirtualBox) = aMachine->i_getVirtualBox();
48 unconst(pSystemProperties) = pVirtualBox->i_getSystemProperties();
49 }
50
51 VirtualBox * const pVirtualBox;
52 SystemProperties * const pSystemProperties;
53 Machine * const pParent;
54 const ComObjPtr<StorageController> pPeer;
55
56 Backupable<settings::StorageController> bd;
57};
58
59
60// constructor / destructor
61/////////////////////////////////////////////////////////////////////////////
62
63DEFINE_EMPTY_CTOR_DTOR(StorageController)
64
65HRESULT StorageController::FinalConstruct()
66{
67 return BaseFinalConstruct();
68}
69
70void StorageController::FinalRelease()
71{
72 uninit();
73 BaseFinalRelease();
74}
75
76// public initializer/uninitializer for internal purposes only
77/////////////////////////////////////////////////////////////////////////////
78
79/**
80 * Initializes the storage controller object.
81 *
82 * @returns COM result indicator.
83 * @param aParent Pointer to our parent object.
84 * @param aName Name of the storage controller.
85 * @param aInstance Instance number of the storage controller.
86 */
87HRESULT StorageController::init(Machine *aParent,
88 const Utf8Str &aName,
89 StorageBus_T aStorageBus,
90 ULONG aInstance, bool fBootable)
91{
92 LogFlowThisFunc(("aParent=%p aName=\"%s\" aInstance=%u\n",
93 aParent, aName.c_str(), aInstance));
94
95 ComAssertRet(aParent && !aName.isEmpty(), E_INVALIDARG);
96 if ( (aStorageBus <= StorageBus_Null)
97 || (aStorageBus > StorageBus_PCIe))
98 return setError(E_INVALIDARG,
99 tr("Invalid storage connection type"));
100
101 ULONG maxInstances;
102 ChipsetType_T chipsetType;
103 HRESULT rc = aParent->COMGETTER(ChipsetType)(&chipsetType);
104 if (FAILED(rc))
105 return rc;
106 rc = aParent->i_getVirtualBox()->i_getSystemProperties()->
107 GetMaxInstancesOfStorageBus(chipsetType, aStorageBus, &maxInstances);
108 if (FAILED(rc))
109 return rc;
110 if (aInstance >= maxInstances)
111 return setError(E_INVALIDARG,
112 tr("Too many storage controllers of this type"));
113
114 /* Enclose the state transition NotReady->InInit->Ready */
115 AutoInitSpan autoInitSpan(this);
116 AssertReturn(autoInitSpan.isOk(), E_FAIL);
117
118 m = new Data(aParent);
119
120 /* m->pPeer is left null */
121
122 m->bd.allocate();
123
124 m->bd->strName = aName;
125 m->bd->ulInstance = aInstance;
126 m->bd->fBootable = fBootable;
127 m->bd->storageBus = aStorageBus;
128 if ( aStorageBus != StorageBus_IDE
129 && aStorageBus != StorageBus_Floppy)
130 m->bd->fUseHostIOCache = false;
131 else
132 m->bd->fUseHostIOCache = true;
133
134 switch (aStorageBus)
135 {
136 case StorageBus_IDE:
137 m->bd->ulPortCount = 2;
138 m->bd->controllerType = StorageControllerType_PIIX4;
139 break;
140 case StorageBus_SATA:
141 m->bd->ulPortCount = 30;
142 m->bd->controllerType = StorageControllerType_IntelAhci;
143 break;
144 case StorageBus_SCSI:
145 m->bd->ulPortCount = 16;
146 m->bd->controllerType = StorageControllerType_LsiLogic;
147 break;
148 case StorageBus_Floppy:
149 m->bd->ulPortCount = 1;
150 m->bd->controllerType = StorageControllerType_I82078;
151 break;
152 case StorageBus_SAS:
153 m->bd->ulPortCount = 8;
154 m->bd->controllerType = StorageControllerType_LsiLogicSas;
155 break;
156 case StorageBus_USB:
157 m->bd->ulPortCount = 8;
158 m->bd->controllerType = StorageControllerType_USB;
159 break;
160 case StorageBus_PCIe:
161 m->bd->ulPortCount = 1;
162 m->bd->controllerType = StorageControllerType_NVMe;
163 break;
164 case StorageBus_Null: break; /* Shut up MSC. */
165 }
166
167 /* Confirm a successful initialization */
168 autoInitSpan.setSucceeded();
169
170 return S_OK;
171}
172
173/**
174 * Initializes the object given another object
175 * (a kind of copy constructor). This object shares data with
176 * the object passed as an argument.
177 *
178 * @param aReshare
179 * When false, the original object will remain a data owner.
180 * Otherwise, data ownership will be transferred from the original
181 * object to this one.
182 *
183 * @note This object must be destroyed before the original object
184 * it shares data with is destroyed.
185 *
186 * @note Locks @a aThat object for writing if @a aReshare is @c true, or for
187 * reading if @a aReshare is false.
188 */
189HRESULT StorageController::init(Machine *aParent,
190 StorageController *aThat,
191 bool aReshare /* = false */)
192{
193 LogFlowThisFunc(("aParent=%p, aThat=%p, aReshare=%RTbool\n",
194 aParent, aThat, aReshare));
195
196 ComAssertRet(aParent && aThat, E_INVALIDARG);
197
198 /* Enclose the state transition NotReady->InInit->Ready */
199 AutoInitSpan autoInitSpan(this);
200 AssertReturn(autoInitSpan.isOk(), E_FAIL);
201
202 m = new Data(aParent);
203
204 /* sanity */
205 AutoCaller thatCaller(aThat);
206 AssertComRCReturnRC(thatCaller.rc());
207
208 if (aReshare)
209 {
210 AutoWriteLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
211
212 unconst(aThat->m->pPeer) = this;
213 m->bd.attach(aThat->m->bd);
214 }
215 else
216 {
217 unconst(m->pPeer) = aThat;
218
219 AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
220 m->bd.share(aThat->m->bd);
221 }
222
223 /* Confirm successful initialization */
224 autoInitSpan.setSucceeded();
225
226 return S_OK;
227}
228
229/**
230 * Initializes the storage controller object given another guest object
231 * (a kind of copy constructor). This object makes a private copy of data
232 * of the original object passed as an argument.
233 */
234HRESULT StorageController::initCopy(Machine *aParent, StorageController *aThat)
235{
236 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
237
238 ComAssertRet(aParent && aThat, E_INVALIDARG);
239
240 /* Enclose the state transition NotReady->InInit->Ready */
241 AutoInitSpan autoInitSpan(this);
242 AssertReturn(autoInitSpan.isOk(), E_FAIL);
243
244 m = new Data(aParent);
245 /* m->pPeer is left null */
246
247 AutoCaller thatCaller(aThat);
248 AssertComRCReturnRC(thatCaller.rc());
249
250 AutoReadLock thatlock(aThat COMMA_LOCKVAL_SRC_POS);
251 m->bd.attachCopy(aThat->m->bd);
252
253 /* Confirm a successful initialization */
254 autoInitSpan.setSucceeded();
255
256 return S_OK;
257}
258
259
260/**
261 * Uninitializes the instance and sets the ready flag to FALSE.
262 * Called either from FinalRelease() or by the parent when it gets destroyed.
263 */
264void StorageController::uninit()
265{
266 LogFlowThisFunc(("\n"));
267
268 /* Enclose the state transition Ready->InUninit->NotReady */
269 AutoUninitSpan autoUninitSpan(this);
270 if (autoUninitSpan.uninitDone())
271 return;
272
273 m->bd.free();
274
275 unconst(m->pPeer) = NULL;
276 unconst(m->pParent) = NULL;
277
278 delete m;
279 m = NULL;
280}
281
282
283// IStorageController properties
284HRESULT StorageController::getName(com::Utf8Str &aName)
285{
286 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
287
288 aName = m->bd->strName;
289
290 return S_OK;
291}
292
293HRESULT StorageController::setName(const com::Utf8Str &aName)
294{
295 /* the machine needs to be mutable */
296 AutoMutableStateDependency adep(m->pParent);
297 if (FAILED(adep.rc())) return adep.rc();
298
299 AutoMultiWriteLock2 alock(m->pParent, this COMMA_LOCKVAL_SRC_POS);
300
301 if (m->bd->strName != aName)
302 {
303 ComObjPtr<StorageController> ctrl;
304 HRESULT rc = m->pParent->i_getStorageControllerByName(aName, ctrl, false /* aSetError */);
305 if (SUCCEEDED(rc))
306 return setError(VBOX_E_OBJECT_IN_USE,
307 tr("Storage controller named '%s' already exists"),
308 aName.c_str());
309
310 Machine::MediaData::AttachmentList atts;
311 rc = m->pParent->i_getMediumAttachmentsOfController(m->bd->strName, atts);
312 for (Machine::MediaData::AttachmentList::const_iterator it = atts.begin();
313 it != atts.end();
314 ++it)
315 {
316 IMediumAttachment *iA = *it;
317 MediumAttachment *pAttach = static_cast<MediumAttachment *>(iA);
318 AutoWriteLock attlock(pAttach COMMA_LOCKVAL_SRC_POS);
319 pAttach->i_updateName(aName);
320 }
321
322
323 m->bd.backup();
324 m->bd->strName = aName;
325
326 m->pParent->i_setModified(Machine::IsModified_Storage);
327 alock.release();
328
329 m->pParent->i_onStorageControllerChange();
330 }
331
332 return S_OK;
333}
334
335HRESULT StorageController::getBus(StorageBus_T *aBus)
336{
337 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
338
339 *aBus = m->bd->storageBus;
340
341 return S_OK;
342}
343
344HRESULT StorageController::getControllerType(StorageControllerType_T *aControllerType)
345{
346 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
347
348 *aControllerType = m->bd->controllerType;
349
350 return S_OK;
351}
352
353HRESULT StorageController::setControllerType(StorageControllerType_T aControllerType)
354{
355 /* the machine needs to be mutable */
356 AutoMutableStateDependency adep(m->pParent);
357 if (FAILED(adep.rc())) return adep.rc();
358
359 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
360
361 HRESULT rc = S_OK;
362
363 switch (m->bd->storageBus)
364 {
365 case StorageBus_IDE:
366 {
367 if ( (aControllerType != StorageControllerType_PIIX3)
368 && (aControllerType != StorageControllerType_PIIX4)
369 && (aControllerType != StorageControllerType_ICH6))
370 rc = E_INVALIDARG;
371 break;
372 }
373 case StorageBus_SATA:
374 {
375 if (aControllerType != StorageControllerType_IntelAhci)
376 rc = E_INVALIDARG;
377 break;
378 }
379 case StorageBus_SCSI:
380 {
381 if ( (aControllerType != StorageControllerType_LsiLogic)
382 && (aControllerType != StorageControllerType_BusLogic))
383 rc = E_INVALIDARG;
384 break;
385 }
386 case StorageBus_Floppy:
387 {
388 if (aControllerType != StorageControllerType_I82078)
389 rc = E_INVALIDARG;
390 break;
391 }
392 case StorageBus_SAS:
393 {
394 if (aControllerType != StorageControllerType_LsiLogicSas)
395 rc = E_INVALIDARG;
396 break;
397 }
398 case StorageBus_USB:
399 {
400 if (aControllerType != StorageControllerType_USB)
401 rc = E_INVALIDARG;
402 break;
403 }
404 case StorageBus_PCIe:
405 {
406 if (aControllerType != StorageControllerType_NVMe)
407 rc = E_INVALIDARG;
408 break;
409 }
410 default:
411 AssertMsgFailed(("Invalid controller type %d\n", m->bd->storageBus));
412 rc = E_INVALIDARG;
413 }
414
415 if (!SUCCEEDED(rc))
416 return setError(rc,
417 tr("Invalid controller type %d"),
418 aControllerType);
419
420 if (m->bd->controllerType != aControllerType)
421 {
422 m->bd.backup();
423 m->bd->controllerType = aControllerType;
424
425 alock.release();
426 AutoWriteLock mlock(m->pParent COMMA_LOCKVAL_SRC_POS); // m->pParent is const, needs no locking
427 m->pParent->i_setModified(Machine::IsModified_Storage);
428 mlock.release();
429
430 m->pParent->i_onStorageControllerChange();
431 }
432
433 return S_OK;
434}
435
436HRESULT StorageController::getMaxDevicesPerPortCount(ULONG *aMaxDevicesPerPortCount)
437{
438 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
439
440 HRESULT rc = m->pSystemProperties->GetMaxDevicesPerPortForStorageBus(m->bd->storageBus, aMaxDevicesPerPortCount);
441
442 return rc;
443}
444
445HRESULT StorageController::getMinPortCount(ULONG *aMinPortCount)
446{
447 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
448
449 HRESULT rc = m->pSystemProperties->GetMinPortCountForStorageBus(m->bd->storageBus, aMinPortCount);
450 return rc;
451}
452
453HRESULT StorageController::getMaxPortCount(ULONG *aMaxPortCount)
454{
455 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
456 HRESULT rc = m->pSystemProperties->GetMaxPortCountForStorageBus(m->bd->storageBus, aMaxPortCount);
457
458 return rc;
459}
460
461HRESULT StorageController::getPortCount(ULONG *aPortCount)
462{
463 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
464
465 *aPortCount = m->bd->ulPortCount;
466
467 return S_OK;
468}
469
470HRESULT StorageController::setPortCount(ULONG aPortCount)
471{
472 /* the machine needs to be mutable */
473 AutoMutableStateDependency adep(m->pParent);
474 if (FAILED(adep.rc())) return adep.rc();
475
476 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
477
478 switch (m->bd->storageBus)
479 {
480 case StorageBus_SATA:
481 {
482 /* AHCI SATA supports a maximum of 30 ports. */
483 if (aPortCount < 1 || aPortCount > 30)
484 return setError(E_INVALIDARG,
485 tr("Invalid port count: %lu (must be in range [%lu, %lu])"),
486 aPortCount, 1, 30);
487 break;
488 }
489 case StorageBus_SCSI:
490 {
491 /*
492 * SCSI does not support setting different ports.
493 * (doesn't make sense here either).
494 * The maximum and minimum is 16 and unless the callee
495 * tries to set a different value we return an error.
496 */
497 if (aPortCount != 16)
498 return setError(E_INVALIDARG,
499 tr("Invalid port count: %lu (must be in range [%lu, %lu])"),
500 aPortCount, 16, 16);
501 break;
502 }
503 case StorageBus_IDE:
504 {
505 /*
506 * The port count is fixed to 2.
507 */
508 if (aPortCount != 2)
509 return setError(E_INVALIDARG,
510 tr("Invalid port count: %lu (must be in range [%lu, %lu])"),
511 aPortCount, 2, 2);
512 break;
513 }
514 case StorageBus_Floppy:
515 {
516 /*
517 * The port count is fixed to 1.
518 */
519 if (aPortCount != 1)
520 return setError(E_INVALIDARG,
521 tr("Invalid port count: %lu (must be in range [%lu, %lu])"),
522 aPortCount, 1, 1);
523 break;
524 }
525 case StorageBus_SAS:
526 {
527 /* SAS supports a maximum of 255 ports. */
528 if (aPortCount < 1 || aPortCount > 255)
529 return setError(E_INVALIDARG,
530 tr("Invalid port count: %lu (must be in range [%lu, %lu])"),
531 aPortCount, 1, 255);
532 break;
533 }
534 case StorageBus_USB:
535 {
536 /*
537 * The port count is fixed to 8.
538 */
539 if (aPortCount != 8)
540 return setError(E_INVALIDARG,
541 tr("Invalid port count: %lu (must be in range [%lu, %lu])"),
542 aPortCount, 8, 8);
543 break;
544 }
545 case StorageBus_PCIe:
546 {
547 /*
548 * PCIe (NVMe in particular) supports theoretically 2^32 - 1
549 * different namespaces, limit the amount artifically here.
550 */
551 if (aPortCount < 1 || aPortCount > 255)
552 return setError(E_INVALIDARG,
553 tr("Invalid port count: %lu (must be in range [%lu, %lu])"),
554 aPortCount, 1, 255);
555 break;
556 }
557 default:
558 AssertMsgFailed(("Invalid controller type %d\n", m->bd->storageBus));
559 }
560
561 if (m->bd->ulPortCount != aPortCount)
562 {
563 m->bd.backup();
564 m->bd->ulPortCount = aPortCount;
565
566 alock.release();
567 AutoWriteLock mlock(m->pParent COMMA_LOCKVAL_SRC_POS); // m->pParent is const, needs no locking
568 m->pParent->i_setModified(Machine::IsModified_Storage);
569 mlock.release();
570
571 m->pParent->i_onStorageControllerChange();
572 }
573
574 return S_OK;
575}
576
577HRESULT StorageController::getInstance(ULONG *aInstance)
578{
579 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
580
581 *aInstance = m->bd->ulInstance;
582
583 return S_OK;
584}
585
586HRESULT StorageController::setInstance(ULONG aInstance)
587{
588 /* the machine needs to be mutable */
589 AutoMutableStateDependency adep(m->pParent);
590 if (FAILED(adep.rc())) return adep.rc();
591
592 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
593
594 if (m->bd->ulInstance != aInstance)
595 {
596 m->bd.backup();
597 m->bd->ulInstance = aInstance;
598
599 alock.release();
600 AutoWriteLock mlock(m->pParent COMMA_LOCKVAL_SRC_POS); // m->pParent is const, needs no locking
601 m->pParent->i_setModified(Machine::IsModified_Storage);
602 mlock.release();
603
604 m->pParent->i_onStorageControllerChange();
605 }
606
607 return S_OK;
608}
609
610HRESULT StorageController::getUseHostIOCache(BOOL *fUseHostIOCache)
611{
612 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
613
614 *fUseHostIOCache = m->bd->fUseHostIOCache;
615
616 return S_OK;
617}
618
619HRESULT StorageController::setUseHostIOCache(BOOL fUseHostIOCache)
620{
621 /* the machine needs to be mutable */
622 AutoMutableStateDependency adep(m->pParent);
623 if (FAILED(adep.rc())) return adep.rc();
624
625 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
626
627 if (m->bd->fUseHostIOCache != !!fUseHostIOCache)
628 {
629 m->bd.backup();
630 m->bd->fUseHostIOCache = !!fUseHostIOCache;
631
632 alock.release();
633 AutoWriteLock mlock(m->pParent COMMA_LOCKVAL_SRC_POS); // m->pParent is const, needs no locking
634 m->pParent->i_setModified(Machine::IsModified_Storage);
635 mlock.release();
636
637 m->pParent->i_onStorageControllerChange();
638 }
639
640 return S_OK;
641}
642
643HRESULT StorageController::getBootable(BOOL *fBootable)
644{
645 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
646
647 *fBootable = m->bd->fBootable;
648
649 return S_OK;
650}
651
652// public methods only for internal purposes
653/////////////////////////////////////////////////////////////////////////////
654
655const Utf8Str& StorageController::i_getName() const
656{
657 return m->bd->strName;
658}
659
660StorageControllerType_T StorageController::i_getControllerType() const
661{
662 return m->bd->controllerType;
663}
664
665StorageBus_T StorageController::i_getStorageBus() const
666{
667 return m->bd->storageBus;
668}
669
670ULONG StorageController::i_getInstance() const
671{
672 return m->bd->ulInstance;
673}
674
675bool StorageController::i_getBootable() const
676{
677 return !!m->bd->fBootable;
678}
679
680/**
681 * Checks the validity of a port and device number.
682 *
683 * @retval S_OK If the given port and device numbers are within the range
684 * supported by this controller.
685 * @retval E_INVALIDARG If not. Sets an error.
686 * @param aControllerPort Controller port number.
687 * @param aDevice Device number.
688 */
689HRESULT StorageController::i_checkPortAndDeviceValid(LONG aControllerPort,
690 LONG aDevice)
691{
692 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
693
694 ULONG portCount = m->bd->ulPortCount;
695 ULONG devicesPerPort;
696 HRESULT rc = m->pSystemProperties->GetMaxDevicesPerPortForStorageBus(m->bd->storageBus, &devicesPerPort);
697 if (FAILED(rc)) return rc;
698
699 if ( aControllerPort < 0
700 || aControllerPort >= (LONG)portCount
701 || aDevice < 0
702 || aDevice >= (LONG)devicesPerPort
703 )
704 return setError(E_INVALIDARG,
705 tr("The port and/or device parameter are out of range: port=%d (must be in range [0, %d]), device=%d (must be in range [0, %d])"),
706 (int)aControllerPort, (int)portCount-1, (int)aDevice, (int)devicesPerPort-1);
707
708 return S_OK;
709}
710
711/** @note Locks objects for writing! */
712void StorageController::i_setBootable(BOOL fBootable)
713{
714 AutoCaller autoCaller(this);
715 AssertComRCReturnVoid(autoCaller.rc());
716
717 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
718
719 m->bd.backup();
720 m->bd->fBootable = RT_BOOL(fBootable);
721}
722
723/** @note Locks objects for writing! */
724void StorageController::i_rollback()
725{
726 AutoCaller autoCaller(this);
727 AssertComRCReturnVoid(autoCaller.rc());
728
729 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
730
731 m->bd.rollback();
732}
733
734/**
735 * @note Locks this object for writing, together with the peer object (also
736 * for writing) if there is one.
737 */
738void StorageController::i_commit()
739{
740 /* sanity */
741 AutoCaller autoCaller(this);
742 AssertComRCReturnVoid(autoCaller.rc());
743
744 /* sanity too */
745 AutoCaller peerCaller(m->pPeer);
746 AssertComRCReturnVoid(peerCaller.rc());
747
748 /* lock both for writing since we modify both (m->pPeer is "master" so locked
749 * first) */
750 AutoMultiWriteLock2 alock(m->pPeer, this COMMA_LOCKVAL_SRC_POS);
751
752 if (m->bd.isBackedUp())
753 {
754 m->bd.commit();
755 if (m->pPeer)
756 {
757 // attach new data to the peer and reshare it
758 m->pPeer->m->bd.attach(m->bd);
759 }
760 }
761}
762
763/**
764 * Cancels sharing (if any) by making an independent copy of data.
765 * This operation also resets this object's peer to NULL.
766 *
767 * @note Locks this object for writing, together with the peer object
768 * represented by @a aThat (locked for reading).
769 */
770void StorageController::i_unshare()
771{
772 /* sanity */
773 AutoCaller autoCaller(this);
774 AssertComRCReturnVoid(autoCaller.rc());
775
776 /* sanity too */
777 AutoCaller peerCaller(m->pPeer);
778 AssertComRCReturnVoid(peerCaller.rc());
779
780 /* peer is not modified, lock it for reading (m->pPeer is "master" so locked
781 * first) */
782 AutoReadLock rl(m->pPeer COMMA_LOCKVAL_SRC_POS);
783 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
784
785 if (m->bd.isShared())
786 {
787 if (!m->bd.isBackedUp())
788 m->bd.backup();
789
790 m->bd.commit();
791 }
792
793 unconst(m->pPeer) = NULL;
794}
795
796Machine* StorageController::i_getMachine()
797{
798 return m->pParent;
799}
800
801ComObjPtr<StorageController> StorageController::i_getPeer()
802{
803 return m->pPeer;
804}
805
806// private methods
807/////////////////////////////////////////////////////////////////////////////
808
809
810/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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