Changeset 24345 in vbox for trunk/src/VBox
- Timestamp:
- Nov 4, 2009 4:50:07 PM (15 years ago)
- svn:sync-xref-src-repo-rev:
- 54367
- Location:
- trunk/src/VBox/Main
- Files:
-
- 6 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Main/MachineImpl.cpp
r24344 r24345 710 710 CheckComRCReturnRC(autoCaller.rc()); 711 711 712 LogFlowThisFunc(("ENTER\n")); 713 712 714 AutoWriteLock alock(this); 713 715 … … 720 722 AutoReinitSpan autoReinitSpan(this); 721 723 AssertReturn(autoReinitSpan.isOk(), E_FAIL); 724 725 #ifdef DEBUG 726 LogFlowThisFunc(("Dumping media backreferences\n")); 727 mParent->dumpAllBackRefs(); 728 #endif 722 729 723 730 if (mData->m_pMachineConfigFile) … … 744 751 if (SUCCEEDED(rc)) 745 752 *aAccessible = mData->mAccessible; 753 754 LogFlowThisFuncLeave(); 746 755 747 756 return rc; … … 5508 5517 data.uuid, 5509 5518 strStateFile); 5510 CheckComRCReturnRC 5519 CheckComRCReturnRC(rc); 5511 5520 5512 5521 /* create a snapshot object */ … … 5543 5552 aCurSnapshotId, 5544 5553 pSnapshot); // parent = the one we created above 5545 CheckComRC BreakRC(rc);5554 CheckComRCReturnRC(rc); 5546 5555 } 5547 5556 … … 5973 5982 } 5974 5983 5975 if ( rc)5984 if (FAILED(rc)) 5976 5985 break; 5977 5986 … … 5994 6003 rc = medium->attachTo(mData->mUuid); 5995 6004 } 5996 AssertComRCBreakRC (rc); 6005 if (FAILED(rc)) 6006 break; 5997 6007 5998 6008 /* backup mMediaData to let registeredInit() properly rollback on failure -
trunk/src/VBox/Main/MediumImpl.cpp
r24273 r24345 68 68 typedef std::list<Guid> GuidList; 69 69 70 BackRef() : inCurState(false) {} 71 72 BackRef(const Guid &aMachineId, const Guid &aSnapshotId = Guid::Empty) 73 : machineId(aMachineId) 74 , inCurState(aSnapshotId.isEmpty()) 70 BackRef(const Guid &aMachineId, 71 const Guid &aSnapshotId = Guid::Empty) 72 : machineId(aMachineId), 73 inCurState(aSnapshotId.isEmpty()) 75 74 { 76 75 if (!aSnapshotId.isEmpty()) 77 snapshotIds.push_back(aSnapshotId);76 llSnapshotIds.push_back(aSnapshotId); 78 77 } 79 78 80 79 Guid machineId; 81 80 bool inCurState : 1; 82 GuidList snapshotIds;81 GuidList llSnapshotIds; 83 82 }; 84 83 … … 1790 1789 if (it->machineId == id) 1791 1790 { 1792 size_t size = it-> snapshotIds.size();1791 size_t size = it->llSnapshotIds.size(); 1793 1792 1794 1793 /* if the medium is attached to the machine in the current state, we … … 1805 1804 it->machineId.toUtf16().detachTo(&snapshotIds [j ++]); 1806 1805 1807 for (BackRef::GuidList::const_iterator jt = 1808 it->snapshotIds.begin();1809 jt != it->snapshotIds.end(); ++ jt, ++j)1806 for (BackRef::GuidList::const_iterator jt = it->llSnapshotIds.begin(); 1807 jt != it->llSnapshotIds.end(); 1808 ++jt, ++j) 1810 1809 { 1811 1810 (*jt).toUtf16().detachTo(&snapshotIds [j]); … … 2653 2652 AssertReturn(!aMachineId.isEmpty(), E_FAIL); 2654 2653 2654 LogFlowThisFunc(("ENTER, aMachineId: {%RTuuid}, aSnapshotId: {%RTuuid}\n", aMachineId.raw(), aSnapshotId.raw())); 2655 2655 2656 AutoCaller autoCaller(this); 2656 2657 AssertComRCReturnRC(autoCaller.rc()); … … 2672 2673 if (m->numCreateDiffTasks > 0) 2673 2674 return setError(E_FAIL, 2674 tr("One or more differencing child hard disks are being created for the hard disk '%s' (%u)"), 2675 m->strLocationFull.raw(), m->numCreateDiffTasks); 2675 tr("Cannot attach hard disk '%s' {%RTuuid}: %u differencing child hard disk(s) are being created"), 2676 m->strLocationFull.raw(), 2677 m->id.raw(), 2678 m->numCreateDiffTasks); 2676 2679 2677 2680 BackRefList::iterator it = … … 2696 2699 2697 2700 /* sanity: no duplicate attachments */ 2698 BackRef::GuidList::const_iterator jt = 2699 std::find(it->snapshotIds.begin(), it->snapshotIds.end(), aSnapshotId); 2700 AssertReturn(jt == it->snapshotIds.end(), E_FAIL); 2701 2702 it->snapshotIds.push_back(aSnapshotId); 2701 for (BackRef::GuidList::const_iterator jt = it->llSnapshotIds.begin(); 2702 jt != it->llSnapshotIds.end(); 2703 ++jt) 2704 { 2705 const Guid &idOldSnapshot = *jt; 2706 2707 if (idOldSnapshot == aSnapshotId) 2708 return setError(E_FAIL, 2709 tr("Cannot attach medium '%s' {%RTuuid} from snapshot '%RTuuid': medium is already in use by snapshot '%RTuuid'"), 2710 m->strLocationFull.raw(), 2711 m->id.raw(), 2712 aSnapshotId.raw(), 2713 idOldSnapshot.raw()); 2714 } 2715 2716 it->llSnapshotIds.push_back(aSnapshotId); 2717 2718 LogFlowThisFuncLeave(); 2703 2719 2704 2720 return S_OK; … … 2737 2753 /* remove the snapshot attachment */ 2738 2754 BackRef::GuidList::iterator jt = 2739 std::find(it-> snapshotIds.begin(), it->snapshotIds.end(), aSnapshotId);2740 2741 AssertReturn(jt != it-> snapshotIds.end(), E_FAIL);2742 it-> snapshotIds.erase(jt);2755 std::find(it->llSnapshotIds.begin(), it->llSnapshotIds.end(), aSnapshotId); 2756 2757 AssertReturn(jt != it->llSnapshotIds.end(), E_FAIL); 2758 it->llSnapshotIds.erase(jt); 2743 2759 } 2744 2760 2745 2761 /* if the backref becomes empty, remove it */ 2746 if (it->inCurState == false && it-> snapshotIds.size() == 0)2762 if (it->inCurState == false && it->llSnapshotIds.size() == 0) 2747 2763 m->backRefs.erase(it); 2748 2764 2749 2765 return S_OK; 2750 2766 } 2767 2768 #ifdef DEBUG 2769 /** 2770 * Debugging helper that gets called after VirtualBox initialization that writes all 2771 * machine backreferences to the debug log. 2772 */ 2773 void Medium::dumpBackRefs() 2774 { 2775 AutoCaller autoCaller(this); 2776 AutoReadLock(this); 2777 2778 LogFlowThisFunc(("Dumping backrefs for medium '%s':\n", m->strLocationFull.raw())); 2779 2780 for (BackRefList::iterator it2 = m->backRefs.begin(); 2781 it2 != m->backRefs.end(); 2782 ++it2) 2783 { 2784 const BackRef &ref = *it2; 2785 LogFlowThisFunc((" Backref from machine {%RTuuid}\n", ref.machineId.raw())); 2786 2787 for (BackRef::GuidList::const_iterator jt2 = it2->llSnapshotIds.begin(); 2788 jt2 != it2->llSnapshotIds.end(); 2789 ++jt2) 2790 { 2791 const Guid &id = *jt2; 2792 LogFlowThisFunc((" Backref from snapshot {%RTuuid}\n", id.raw())); 2793 } 2794 } 2795 } 2796 #endif 2751 2797 2752 2798 /** … … 2787 2833 2788 2834 /** 2835 * Internal method to return the medium's size. Must have caller + locking! 2836 * @return 2837 */ 2838 uint64_t Medium::size() const 2839 { 2840 return m->size; 2841 } 2842 2843 /** 2789 2844 * Internal method to return the medium's list of backrefs. Must have caller + locking! 2790 2845 * @return 2791 2846 */ 2792 // const Medium::BackRefList& Medium::backRefs() const2793 // {2794 // return m->backRefs;2795 // }2796 2797 2847 const Guid* Medium::getFirstMachineBackrefId() const 2798 2848 { … … 2809 2859 2810 2860 const BackRef &ref = m->backRefs.front(); 2811 if (!ref. snapshotIds.size())2861 if (!ref.llSnapshotIds.size()) 2812 2862 return NULL; 2813 2863 2814 return &ref. snapshotIds.front();2864 return &ref.llSnapshotIds.front(); 2815 2865 } 2816 2866 … … 2934 2984 for (BackRefList::const_iterator it = m->backRefs.begin(); 2935 2985 it != m->backRefs.end(); ++ it) 2936 if (it-> snapshotIds.size() != 0)2986 if (it->llSnapshotIds.size() != 0) 2937 2987 return true; 2938 2988 … … 3139 3189 * time when discard() is called, there must be no any attachments at all 3140 3190 * (the code calling prepareDiscard() should detach). */ 3141 AssertReturn(m->backRefs.size() == 1 && 3142 !m->backRefs.front().inCurState && 3143 m->backRefs.front().snapshotIds.size() == 1, E_FAIL); 3191 AssertReturn( m->backRefs.size() == 1 3192 && !m->backRefs.front().inCurState 3193 && m->backRefs.front().llSnapshotIds.size() == 1, 3194 E_FAIL); 3144 3195 3145 3196 ComObjPtr<Medium> child = children().front(); … … 3215 3266 * reading or writing. 3216 3267 */ 3217 HRESULT Medium::discard(ComObjPtr<Progress> &aProgress, MergeChain *aChain)3268 HRESULT Medium::discard(ComObjPtr<Progress> &aProgress, ULONG ulWeight, MergeChain *aChain) 3218 3269 { 3219 3270 AssertReturn(!aProgress.isNull(), E_FAIL); … … 3228 3279 3229 3280 aProgress->SetNextOperation(BstrFmt(tr("Discarding hard disk '%s'"), name().raw()), 3230 1); // weight3281 ulWeight); // weight 3231 3282 3232 3283 if (aChain == NULL) … … 4175 4226 * this legal situatinon and do not report an error. */ 4176 4227 4177 if (it-> snapshotIds.size() == 0)4228 if (it->llSnapshotIds.size() == 0) 4178 4229 { 4179 4230 return setError(VBOX_E_INVALID_OBJECT_STATE, … … 4182 4233 } 4183 4234 4184 Assert(it-> snapshotIds.size() == 1);4235 Assert(it->llSnapshotIds.size() == 1); 4185 4236 } 4186 4237 } -
trunk/src/VBox/Main/SnapshotImpl.cpp
r24301 r24345 1506 1506 1507 1507 ComObjPtr<Snapshot> pSnapshot(static_cast<Snapshot*>(aSnapshot)); 1508 ComPtr<SnapshotMachine> pSnapMachine = pSnapshot->getSnapshotMachine(); 1508 1509 1509 1510 // create a progress object. The number of operations is: 1510 1511 // 1 (preparing) + # of hard disks + 1 (if we need to copy the saved state file) */ 1511 ComObjPtr<Progress> progress;1512 progress.createObject();1513 1514 1512 LogFlowThisFunc(("Going thru snapshot machine attachments to determine progress setup\n")); 1515 1513 1516 1514 ULONG ulOpCount = 1; // one for preparations 1517 1515 ULONG ulTotalWeight = 1; // one for preparations 1518 for (MediaData::AttachmentList::iterator it = pSnap shot->getSnapshotMachine()->mMediaData->mAttachments.begin();1519 it != pSnap shot->getSnapshotMachine()->mMediaData->mAttachments.end();1516 for (MediaData::AttachmentList::iterator it = pSnapMachine->mMediaData->mAttachments.begin(); 1517 it != pSnapMachine->mMediaData->mAttachments.end(); 1520 1518 ++it) 1521 1519 { … … 1551 1549 } 1552 1550 1553 progress->init(mParent, aInitiator, 1554 Bstr(tr("Restoring snapshot")), 1555 FALSE /* aCancelable */, 1556 ulOpCount, 1557 ulTotalWeight, 1558 Bstr(tr("Restoring machine settings")), 1559 1); 1551 ComObjPtr<Progress> pProgress; 1552 pProgress.createObject(); 1553 pProgress->init(mParent, aInitiator, 1554 BstrFmt(tr("Restoring snapshot '%s'"), pSnapshot->getName().c_str()), 1555 FALSE /* aCancelable */, 1556 ulOpCount, 1557 ulTotalWeight, 1558 Bstr(tr("Restoring machine settings")), 1559 1); 1560 1560 1561 1561 /* create and start the task on a separate thread (note that it will not 1562 1562 * start working until we release alock) */ 1563 1563 RestoreSnapshotTask *task = new RestoreSnapshotTask(this, 1564 p rogress,1564 pProgress, 1565 1565 pSnapshot, 1566 1566 ulStateFileSizeMB); … … 1582 1582 1583 1583 /* return the progress to the caller */ 1584 p rogress.queryInterfaceTo(aProgress);1584 pProgress.queryInterfaceTo(aProgress); 1585 1585 1586 1586 /* return the new state to the caller */ … … 1909 1909 AutoWriteLock treeLock(snapshotsTreeLockHandle()); 1910 1910 1911 ComObjPtr<Snapshot> snapshot;1912 HRESULT rc = findSnapshot(id, snapshot, true /* aSetError */);1911 ComObjPtr<Snapshot> pSnapshot; 1912 HRESULT rc = findSnapshot(id, pSnapshot, true /* aSetError */); 1913 1913 CheckComRCReturnRC(rc); 1914 1914 1915 AutoWriteLock snapshotLock( snapshot);1916 1917 size_t childrenCount = snapshot->getChildrenCount();1915 AutoWriteLock snapshotLock(pSnapshot); 1916 1917 size_t childrenCount = pSnapshot->getChildrenCount(); 1918 1918 if (childrenCount > 1) 1919 1919 return setError(VBOX_E_INVALID_OBJECT_STATE, 1920 1920 tr("Snapshot '%s' of the machine '%ls' cannot be deleted. because it has %d child snapshots, which is more than the one snapshot allowed for deletion"), 1921 snapshot->getName().c_str(),1921 pSnapshot->getName().c_str(), 1922 1922 mUserData->mName.raw(), 1923 1923 childrenCount); … … 1926 1926 * settings are committed and saved. 1927 1927 */ 1928 if ( snapshot == mData->mCurrentSnapshot)1928 if (pSnapshot == mData->mCurrentSnapshot) 1929 1929 { 1930 1930 if (isModified()) … … 1935 1935 } 1936 1936 1937 ComPtr<SnapshotMachine> pSnapMachine = pSnapshot->getSnapshotMachine(); 1938 1937 1939 /* create a progress object. The number of operations is: 1938 * 1 (preparing) + # of hard disks + 1 if the snapshot is online1940 * 1 (preparing) + 1 if the snapshot is online + # of normal hard disks 1939 1941 */ 1940 ComObjPtr<Progress> progress; 1941 progress.createObject(); 1942 rc = progress->init(mParent, aInitiator, 1943 BstrFmt(tr("Discarding snapshot '%s'"), 1944 snapshot->getName().c_str()), 1945 FALSE /* aCancelable */, 1946 1 + (ULONG)snapshot->getSnapshotMachine()->mMediaData->mAttachments.size() 1947 + (snapshot->stateFilePath().length() ? 1 : 0), 1948 Bstr(tr("Preparing to discard snapshot"))); 1949 AssertComRCReturn(rc, rc); 1942 LogFlowThisFunc(("Going thru snapshot machine attachments to determine progress setup\n")); 1943 1944 ULONG ulOpCount = 1; // one for preparations 1945 ULONG ulTotalWeight = 1; // one for preparations 1946 1947 if (pSnapshot->stateFilePath().length()) 1948 { 1949 ++ulOpCount; 1950 ++ulTotalWeight; // assume 1 MB for deleting the state file 1951 } 1952 1953 // count normal hard disks and add their sizes to the weight 1954 for (MediaData::AttachmentList::iterator it = pSnapMachine->mMediaData->mAttachments.begin(); 1955 it != pSnapMachine->mMediaData->mAttachments.end(); 1956 ++it) 1957 { 1958 ComObjPtr<MediumAttachment> &pAttach = *it; 1959 AutoReadLock attachLock(pAttach); 1960 if (pAttach->type() == DeviceType_HardDisk) 1961 { 1962 Assert(pAttach->medium()); 1963 ComObjPtr<Medium> pHD = pAttach->medium(); 1964 AutoReadLock mlock(pHD); 1965 if (pHD->type() == MediumType_Normal) 1966 { 1967 ++ulOpCount; 1968 ulTotalWeight += pHD->size() / _1M; 1969 } 1970 LogFlowThisFunc(("op %d: considering hard disk attachment %s\n", ulOpCount, pHD->name().c_str())); 1971 } 1972 } 1973 1974 ComObjPtr<Progress> pProgress; 1975 pProgress.createObject(); 1976 pProgress->init(mParent, aInitiator, 1977 BstrFmt(tr("Deleting snapshot '%s'"), pSnapshot->getName().c_str()), 1978 FALSE /* aCancelable */, 1979 ulOpCount, 1980 ulTotalWeight, 1981 Bstr(tr("Setting up")), 1982 1); 1950 1983 1951 1984 /* create and start the task on a separate thread */ 1952 DeleteSnapshotTask *task = new DeleteSnapshotTask(this, p rogress, snapshot);1985 DeleteSnapshotTask *task = new DeleteSnapshotTask(this, pProgress, pSnapshot); 1953 1986 int vrc = RTThreadCreate(NULL, 1954 1987 taskHandler, … … 1968 2001 1969 2002 /* return the progress to the caller */ 1970 p rogress.queryInterfaceTo(aProgress);2003 pProgress.queryInterfaceTo(aProgress); 1971 2004 1972 2005 /* return the new state to the caller */ … … 2051 2084 aTask.pSnapshot->lockHandle()); 2052 2085 2053 ComPtr<SnapshotMachine> sm= aTask.pSnapshot->getSnapshotMachine();2086 ComPtr<SnapshotMachine> pSnapMachine = aTask.pSnapshot->getSnapshotMachine(); 2054 2087 /* no need to lock the snapshot machine since it is const by definiton */ 2055 2088 … … 2068 2101 LogFlowThisFunc(("1: Checking hard disk merge prerequisites...\n")); 2069 2102 2070 for (MediaData::AttachmentList:: const_iterator it = sm->mMediaData->mAttachments.begin();2071 it != sm->mMediaData->mAttachments.end();2072 2103 for (MediaData::AttachmentList::iterator it = pSnapMachine->mMediaData->mAttachments.begin(); 2104 it != pSnapMachine->mMediaData->mAttachments.end(); 2105 ++it) 2073 2106 { 2074 ComObjPtr<MediumAttachment> hda = *it; 2075 ComObjPtr<Medium> hd = hda->medium(); 2076 2077 // medium can be NULL only for non-hard-disk types 2078 Assert( !hd.isNull() 2079 || hda->type() != DeviceType_HardDisk); 2080 if (hd.isNull()) 2081 continue; 2082 2083 /* Medium::prepareDiscard() reqiuires a write lock */ 2084 AutoWriteLock hdLock(hd); 2085 2086 if (hd->type() != MediumType_Normal) 2107 ComObjPtr<MediumAttachment> &pAttach = *it; 2108 AutoReadLock attachLock(pAttach); 2109 if (pAttach->type() == DeviceType_HardDisk) 2087 2110 { 2088 /* skip writethrough hard disks */ 2089 Assert(hd->type() == MediumType_Writethrough); 2090 rc = aTask.pProgress->SetNextOperation(BstrFmt(tr("Skipping writethrough hard disk '%s'"), 2091 hd->base()->name().raw()), 2092 1); // weight 2111 Assert(pAttach->medium()); 2112 ComObjPtr<Medium> pHD = pAttach->medium(); 2113 AutoReadLock mlock(pHD); 2114 2115 Medium::MergeChain *chain = NULL; 2116 2117 /* needs to be discarded (merged with the child if any), check 2118 * prerequisites */ 2119 rc = pHD->prepareDiscard(chain); 2093 2120 CheckComRCThrowRC(rc); 2094 continue; 2121 2122 if (pHD->parent().isNull() && chain != NULL) 2123 { 2124 /* it's a base hard disk so it will be a backward merge of its 2125 * only child to it (prepareDiscard() does necessary checks). We 2126 * need then to update the attachment that refers to the child 2127 * to refer to the parent instead. Don't forget to detach the 2128 * child (otherwise mergeTo() called by discard() will assert 2129 * because it will be going to delete the child) */ 2130 2131 /* The below assert would be nice but I don't want to move 2132 * Medium::MergeChain to the header just for that 2133 * Assert (!chain->isForward()); */ 2134 2135 Assert(pHD->children().size() == 1); 2136 2137 ComObjPtr<Medium> replaceHd = pHD->children().front(); 2138 2139 const Guid *pReplaceMachineId = replaceHd->getFirstMachineBackrefId(); 2140 Assert(pReplaceMachineId && *pReplaceMachineId == mData->mUuid); 2141 2142 Guid snapshotId; 2143 const Guid *pSnapshotId = replaceHd->getFirstMachineBackrefSnapshotId(); 2144 if (pSnapshotId) 2145 snapshotId = *pSnapshotId; 2146 2147 HRESULT rc2 = S_OK; 2148 2149 /* adjust back references */ 2150 rc2 = replaceHd->detachFrom(mData->mUuid, snapshotId); 2151 AssertComRC(rc2); 2152 2153 rc2 = pHD->attachTo(mData->mUuid, snapshotId); 2154 AssertComRC(rc2); 2155 2156 /* replace the hard disk in the attachment object */ 2157 if (snapshotId.isEmpty()) 2158 { 2159 /* in current state */ 2160 AssertBreak(pAttach = findAttachment(mMediaData->mAttachments, replaceHd)); 2161 } 2162 else 2163 { 2164 /* in snapshot */ 2165 ComObjPtr<Snapshot> snapshot; 2166 rc2 = findSnapshot(snapshotId, snapshot); 2167 AssertComRC(rc2); 2168 2169 /* don't lock the snapshot; cannot be modified outside */ 2170 MediaData::AttachmentList &snapAtts = snapshot->getSnapshotMachine()->mMediaData->mAttachments; 2171 AssertBreak(pAttach = findAttachment(snapAtts, replaceHd)); 2172 } 2173 2174 attachLock.unlock(); 2175 AutoWriteLock attLock(pAttach); 2176 pAttach->updateMedium(pHD, false /* aImplicit */); 2177 2178 toDiscard.push_back(MediumDiscardRec(pHD, 2179 chain, 2180 replaceHd, 2181 pAttach, 2182 snapshotId)); 2183 continue; 2184 } 2185 2186 toDiscard.push_back(MediumDiscardRec(pHD, chain)); 2095 2187 } 2096 2097 Medium::MergeChain *chain = NULL;2098 2099 /* needs to be discarded (merged with the child if any), check2100 * prerequisites */2101 rc = hd->prepareDiscard(chain);2102 CheckComRCThrowRC(rc);2103 2104 if (hd->parent().isNull() && chain != NULL)2105 {2106 /* it's a base hard disk so it will be a backward merge of its2107 * only child to it (prepareDiscard() does necessary checks). We2108 * need then to update the attachment that refers to the child2109 * to refer to the parent instead. Don't forget to detach the2110 * child (otherwise mergeTo() called by discard() will assert2111 * because it will be going to delete the child) */2112 2113 /* The below assert would be nice but I don't want to move2114 * Medium::MergeChain to the header just for that2115 * Assert (!chain->isForward()); */2116 2117 Assert(hd->children().size() == 1);2118 2119 ComObjPtr<Medium> replaceHd = hd->children().front();2120 2121 const Guid *pReplaceMachineId = replaceHd->getFirstMachineBackrefId();2122 Assert(pReplaceMachineId && *pReplaceMachineId == mData->mUuid);2123 2124 Guid snapshotId;2125 const Guid *pSnapshotId = replaceHd->getFirstMachineBackrefSnapshotId();2126 if (pSnapshotId)2127 snapshotId = *pSnapshotId;2128 2129 HRESULT rc2 = S_OK;2130 2131 /* adjust back references */2132 rc2 = replaceHd->detachFrom (mData->mUuid, snapshotId);2133 AssertComRC(rc2);2134 2135 rc2 = hd->attachTo (mData->mUuid, snapshotId);2136 AssertComRC(rc2);2137 2138 /* replace the hard disk in the attachment object */2139 if (snapshotId.isEmpty())2140 {2141 /* in current state */2142 AssertBreak(hda = findAttachment(mMediaData->mAttachments, replaceHd));2143 }2144 else2145 {2146 /* in snapshot */2147 ComObjPtr<Snapshot> snapshot;2148 rc2 = findSnapshot(snapshotId, snapshot);2149 AssertComRC(rc2);2150 2151 /* don't lock the snapshot; cannot be modified outside */2152 MediaData::AttachmentList &snapAtts = snapshot->getSnapshotMachine()->mMediaData->mAttachments;2153 AssertBreak(hda = findAttachment(snapAtts, replaceHd));2154 }2155 2156 AutoWriteLock attLock(hda);2157 hda->updateMedium(hd, false /* aImplicit */);2158 2159 toDiscard.push_back(MediumDiscardRec(hd,2160 chain,2161 replaceHd,2162 hda,2163 snapshotId));2164 continue;2165 }2166 2167 toDiscard.push_back(MediumDiscardRec(hd, chain));2168 2188 } 2169 2189 … … 2229 2249 it != toDiscard.end();) 2230 2250 { 2231 rc = it->hd->discard(aTask.pProgress, it->chain); 2251 rc = it->hd->discard(aTask.pProgress, 2252 it->hd->size() / _1M, // weight 2253 it->chain); 2232 2254 CheckComRCBreakRC(rc); 2233 2255 2234 2256 /* prevent from calling cancelDiscard() */ 2235 it = toDiscard.erase 2257 it = toDiscard.erase(it); 2236 2258 } 2237 2259 2260 LogFlowThisFunc(("Entering locks again...\n")); 2238 2261 alock.enter(); 2262 LogFlowThisFunc(("Entered locks OK\n")); 2239 2263 2240 2264 CheckComRCThrowRC(rc); -
trunk/src/VBox/Main/VirtualBoxImpl.cpp
r24298 r24345 389 389 throw rc; 390 390 391 392 #ifdef DEBUG 393 LogFlowThisFunc(("Dumping media backreferences\n")); 394 dumpAllBackRefs(); 395 #endif 396 391 397 /* net services */ 392 398 for (settings::DHCPServersList::const_iterator it = m->pMainConfigFile->llDhcpServers.begin(); … … 1908 1914 // public methods only for internal purposes 1909 1915 ///////////////////////////////////////////////////////////////////////////// 1916 1917 #ifdef DEBUG 1918 void VirtualBox::dumpAllBackRefs() 1919 { 1920 for (HardDiskList::const_iterator mt = m->llHardDisks.begin(); 1921 mt != m->llHardDisks.end(); 1922 ++mt) 1923 { 1924 ComObjPtr<Medium> pMedium = *mt; 1925 pMedium->dumpBackRefs(); 1926 } 1927 for (DVDImageList::const_iterator mt = m->llDVDImages.begin(); 1928 mt != m->llDVDImages.end(); 1929 ++mt) 1930 { 1931 ComObjPtr<Medium> pMedium = *mt; 1932 pMedium->dumpBackRefs(); 1933 } 1934 } 1935 #endif 1910 1936 1911 1937 /** … … 3274 3300 * 3275 3301 * @param aHardDisk Hard disk object to remember. 3276 * @param aSaveRegistry @c true to save hard disk registry to disk (default).3302 * @param aSaveRegistry Whether to save the registry after adding the new disk; this is @c false when this gets called during VirtualBox initialization. 3277 3303 * 3278 3304 * When @a aSaveRegistry is @c true, this operation may fail because of the -
trunk/src/VBox/Main/include/MediumImpl.h
r24273 r24345 163 163 const Guid &aSnapshotId = Guid::Empty); 164 164 165 #ifdef DEBUG 166 void dumpBackRefs(); 167 #endif 168 165 169 const Guid& id() const; 166 170 MediumState_T state() const; 167 171 const Utf8Str& location() const; 168 172 const Utf8Str& locationFull() const; 173 uint64_t size() const; 169 174 170 175 const Guid* getFirstMachineBackrefId() const; … … 244 249 245 250 HRESULT prepareDiscard(MergeChain * &aChain); 246 HRESULT discard(ComObjPtr<Progress> &aProgress, MergeChain *aChain);251 HRESULT discard(ComObjPtr<Progress> &aProgress, ULONG ulWeight, MergeChain *aChain); 247 252 void cancelDiscard(MergeChain *aChain); 248 253 -
trunk/src/VBox/Main/include/VirtualBoxImpl.h
r24298 r24345 78 78 friend class CallbackEvent; 79 79 80 VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT 80 VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT(VirtualBox) 81 81 82 82 DECLARE_CLASSFACTORY_SINGLETON(VirtualBox) … … 179 179 180 180 /* public methods only for internal purposes */ 181 #ifdef DEBUG 182 void dumpAllBackRefs(); 183 #endif 181 184 182 185 HRESULT postEvent(Event *event);
Note:
See TracChangeset
for help on using the changeset viewer.