VirtualBox

Changeset 24345 in vbox for trunk/src/VBox


Ignore:
Timestamp:
Nov 4, 2009 4:50:07 PM (15 years ago)
Author:
vboxsync
svn:sync-xref-src-repo-rev:
54367
Message:

Main: make backrefs code a bit more readable + add backrefs logging, fix deleteSnapshot() progress bars, but deleteSnapshot() is still broken

Location:
trunk/src/VBox/Main
Files:
6 edited

Legend:

Unmodified
Added
Removed
  • trunk/src/VBox/Main/MachineImpl.cpp

    r24344 r24345  
    710710    CheckComRCReturnRC(autoCaller.rc());
    711711
     712    LogFlowThisFunc(("ENTER\n"));
     713
    712714    AutoWriteLock alock(this);
    713715
     
    720722        AutoReinitSpan autoReinitSpan(this);
    721723        AssertReturn(autoReinitSpan.isOk(), E_FAIL);
     724
     725#ifdef DEBUG
     726        LogFlowThisFunc(("Dumping media backreferences\n"));
     727        mParent->dumpAllBackRefs();
     728#endif
    722729
    723730        if (mData->m_pMachineConfigFile)
     
    744751    if (SUCCEEDED(rc))
    745752        *aAccessible = mData->mAccessible;
     753
     754    LogFlowThisFuncLeave();
    746755
    747756    return rc;
     
    55085517                                data.uuid,
    55095518                                strStateFile);
    5510     CheckComRCReturnRC (rc);
     5519    CheckComRCReturnRC(rc);
    55115520
    55125521    /* create a snapshot object */
     
    55435552                          aCurSnapshotId,
    55445553                          pSnapshot);       // parent = the one we created above
    5545         CheckComRCBreakRC(rc);
     5554        CheckComRCReturnRC(rc);
    55465555    }
    55475556
     
    59735982        }
    59745983
    5975         if (rc)
     5984        if (FAILED(rc))
    59765985            break;
    59775986
     
    59946003                rc = medium->attachTo(mData->mUuid);
    59956004        }
    5996         AssertComRCBreakRC (rc);
     6005        if (FAILED(rc))
     6006            break;
    59976007
    59986008        /* backup mMediaData to let registeredInit() properly rollback on failure
  • trunk/src/VBox/Main/MediumImpl.cpp

    r24273 r24345  
    6868    typedef std::list<Guid> GuidList;
    6969
    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())
    7574    {
    7675        if (!aSnapshotId.isEmpty())
    77             snapshotIds.push_back(aSnapshotId);
     76            llSnapshotIds.push_back(aSnapshotId);
    7877    }
    7978
    8079    Guid machineId;
    8180    bool inCurState : 1;
    82     GuidList snapshotIds;
     81    GuidList llSnapshotIds;
    8382};
    8483
     
    17901789        if (it->machineId == id)
    17911790        {
    1792             size_t size = it->snapshotIds.size();
     1791            size_t size = it->llSnapshotIds.size();
    17931792
    17941793            /* if the medium is attached to the machine in the current state, we
     
    18051804                    it->machineId.toUtf16().detachTo(&snapshotIds [j ++]);
    18061805
    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)
    18101809                {
    18111810                     (*jt).toUtf16().detachTo(&snapshotIds [j]);
     
    26532652    AssertReturn(!aMachineId.isEmpty(), E_FAIL);
    26542653
     2654    LogFlowThisFunc(("ENTER, aMachineId: {%RTuuid}, aSnapshotId: {%RTuuid}\n", aMachineId.raw(), aSnapshotId.raw()));
     2655
    26552656    AutoCaller autoCaller(this);
    26562657    AssertComRCReturnRC(autoCaller.rc());
     
    26722673    if (m->numCreateDiffTasks > 0)
    26732674        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);
    26762679
    26772680    BackRefList::iterator it =
     
    26962699
    26972700    /* 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();
    27032719
    27042720    return S_OK;
     
    27372753        /* remove the snapshot attachment */
    27382754        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);
    27432759    }
    27442760
    27452761    /* 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)
    27472763        m->backRefs.erase(it);
    27482764
    27492765    return S_OK;
    27502766}
     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 */
     2773void 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
    27512797
    27522798/**
     
    27872833
    27882834/**
     2835 * Internal method to return the medium's size. Must have caller + locking!
     2836 * @return
     2837 */
     2838uint64_t Medium::size() const
     2839{
     2840    return m->size;
     2841}
     2842
     2843/**
    27892844 * Internal method to return the medium's list of backrefs. Must have caller + locking!
    27902845 * @return
    27912846 */
    2792 // const Medium::BackRefList& Medium::backRefs() const
    2793 // {
    2794 //     return m->backRefs;
    2795 // }
    2796 
    27972847const Guid* Medium::getFirstMachineBackrefId() const
    27982848{
     
    28092859
    28102860    const BackRef &ref = m->backRefs.front();
    2811     if (!ref.snapshotIds.size())
     2861    if (!ref.llSnapshotIds.size())
    28122862        return NULL;
    28132863
    2814     return &ref.snapshotIds.front();
     2864    return &ref.llSnapshotIds.front();
    28152865}
    28162866
     
    29342984            for (BackRefList::const_iterator it = m->backRefs.begin();
    29352985                 it != m->backRefs.end(); ++ it)
    2936                 if (it->snapshotIds.size() != 0)
     2986                if (it->llSnapshotIds.size() != 0)
    29372987                    return true;
    29382988
     
    31393189     * time when discard() is called, there must be no any attachments at all
    31403190     * (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);
    31443195
    31453196    ComObjPtr<Medium> child = children().front();
     
    32153266 *       reading or writing.
    32163267 */
    3217 HRESULT Medium::discard(ComObjPtr<Progress> &aProgress, MergeChain *aChain)
     3268HRESULT Medium::discard(ComObjPtr<Progress> &aProgress, ULONG ulWeight, MergeChain *aChain)
    32183269{
    32193270    AssertReturn(!aProgress.isNull(), E_FAIL);
     
    32283279
    32293280        aProgress->SetNextOperation(BstrFmt(tr("Discarding hard disk '%s'"), name().raw()),
    3230                                     1);        // weight
     3281                                    ulWeight);        // weight
    32313282
    32323283        if (aChain == NULL)
     
    41754226             * this legal situatinon and do not report an error. */
    41764227
    4177             if (it->snapshotIds.size() == 0)
     4228            if (it->llSnapshotIds.size() == 0)
    41784229            {
    41794230                return setError(VBOX_E_INVALID_OBJECT_STATE,
     
    41824233            }
    41834234
    4184             Assert(it->snapshotIds.size() == 1);
     4235            Assert(it->llSnapshotIds.size() == 1);
    41854236        }
    41864237    }
  • trunk/src/VBox/Main/SnapshotImpl.cpp

    r24301 r24345  
    15061506
    15071507    ComObjPtr<Snapshot> pSnapshot(static_cast<Snapshot*>(aSnapshot));
     1508    ComPtr<SnapshotMachine> pSnapMachine = pSnapshot->getSnapshotMachine();
    15081509
    15091510    // create a progress object. The number of operations is:
    15101511    // 1 (preparing) + # of hard disks + 1 (if we need to copy the saved state file) */
    1511     ComObjPtr<Progress> progress;
    1512     progress.createObject();
    1513 
    15141512    LogFlowThisFunc(("Going thru snapshot machine attachments to determine progress setup\n"));
    15151513
    15161514    ULONG ulOpCount = 1;            // one for preparations
    15171515    ULONG ulTotalWeight = 1;        // one for preparations
    1518     for (MediaData::AttachmentList::iterator it = pSnapshot->getSnapshotMachine()->mMediaData->mAttachments.begin();
    1519          it != pSnapshot->getSnapshotMachine()->mMediaData->mAttachments.end();
     1516    for (MediaData::AttachmentList::iterator it = pSnapMachine->mMediaData->mAttachments.begin();
     1517         it != pSnapMachine->mMediaData->mAttachments.end();
    15201518         ++it)
    15211519    {
     
    15511549    }
    15521550
    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);
    15601560
    15611561    /* create and start the task on a separate thread (note that it will not
    15621562     * start working until we release alock) */
    15631563    RestoreSnapshotTask *task = new RestoreSnapshotTask(this,
    1564                                                         progress,
     1564                                                        pProgress,
    15651565                                                        pSnapshot,
    15661566                                                        ulStateFileSizeMB);
     
    15821582
    15831583    /* return the progress to the caller */
    1584     progress.queryInterfaceTo(aProgress);
     1584    pProgress.queryInterfaceTo(aProgress);
    15851585
    15861586    /* return the new state to the caller */
     
    19091909    AutoWriteLock treeLock(snapshotsTreeLockHandle());
    19101910
    1911     ComObjPtr<Snapshot> snapshot;
    1912     HRESULT rc = findSnapshot(id, snapshot, true /* aSetError */);
     1911    ComObjPtr<Snapshot> pSnapshot;
     1912    HRESULT rc = findSnapshot(id, pSnapshot, true /* aSetError */);
    19131913    CheckComRCReturnRC(rc);
    19141914
    1915     AutoWriteLock snapshotLock(snapshot);
    1916 
    1917     size_t childrenCount = snapshot->getChildrenCount();
     1915    AutoWriteLock snapshotLock(pSnapshot);
     1916
     1917    size_t childrenCount = pSnapshot->getChildrenCount();
    19181918    if (childrenCount > 1)
    19191919        return setError(VBOX_E_INVALID_OBJECT_STATE,
    19201920                        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(),
    19221922                        mUserData->mName.raw(),
    19231923                        childrenCount);
     
    19261926     * settings are committed and saved.
    19271927     */
    1928     if (snapshot == mData->mCurrentSnapshot)
     1928    if (pSnapshot == mData->mCurrentSnapshot)
    19291929    {
    19301930        if (isModified())
     
    19351935    }
    19361936
     1937    ComPtr<SnapshotMachine> pSnapMachine = pSnapshot->getSnapshotMachine();
     1938
    19371939    /* create a progress object. The number of operations is:
    1938      *   1 (preparing) + # of hard disks + 1 if the snapshot is online
     1940     *   1 (preparing) + 1 if the snapshot is online + # of normal hard disks
    19391941     */
    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);
    19501983
    19511984    /* create and start the task on a separate thread */
    1952     DeleteSnapshotTask *task = new DeleteSnapshotTask(this, progress, snapshot);
     1985    DeleteSnapshotTask *task = new DeleteSnapshotTask(this, pProgress, pSnapshot);
    19531986    int vrc = RTThreadCreate(NULL,
    19541987                             taskHandler,
     
    19682001
    19692002    /* return the progress to the caller */
    1970     progress.queryInterfaceTo(aProgress);
     2003    pProgress.queryInterfaceTo(aProgress);
    19712004
    19722005    /* return the new state to the caller */
     
    20512084                              aTask.pSnapshot->lockHandle());
    20522085
    2053     ComPtr<SnapshotMachine> sm = aTask.pSnapshot->getSnapshotMachine();
     2086    ComPtr<SnapshotMachine> pSnapMachine = aTask.pSnapshot->getSnapshotMachine();
    20542087    /* no need to lock the snapshot machine since it is const by definiton */
    20552088
     
    20682101        LogFlowThisFunc(("1: Checking hard disk merge prerequisites...\n"));
    20692102
    2070         for (MediaData::AttachmentList::const_iterator it = sm->mMediaData->mAttachments.begin();
    2071              it != sm->mMediaData->mAttachments.end();
    2072              ++it)
     2103        for (MediaData::AttachmentList::iterator it = pSnapMachine->mMediaData->mAttachments.begin();
     2104            it != pSnapMachine->mMediaData->mAttachments.end();
     2105            ++it)
    20732106        {
    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)
    20872110            {
    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);
    20932120                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));
    20952187            }
    2096 
    2097             Medium::MergeChain *chain = NULL;
    2098 
    2099             /* needs to be discarded (merged with the child if any), check
    2100              * 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 its
    2107                  * only child to it (prepareDiscard() does necessary checks). We
    2108                  * need then to update the attachment that refers to the child
    2109                  * to refer to the parent instead. Don't forget to detach the
    2110                  * child (otherwise mergeTo() called by discard() will assert
    2111                  * because it will be going to delete the child) */
    2112 
    2113                 /* The below assert would be nice but I don't want to move
    2114                  * Medium::MergeChain to the header just for that
    2115                  * 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                 else
    2145                 {
    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));
    21682188        }
    21692189
     
    22292249             it != toDiscard.end();)
    22302250        {
    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);
    22322254            CheckComRCBreakRC(rc);
    22332255
    22342256            /* prevent from calling cancelDiscard() */
    2235             it = toDiscard.erase (it);
     2257            it = toDiscard.erase(it);
    22362258        }
    22372259
     2260        LogFlowThisFunc(("Entering locks again...\n"));
    22382261        alock.enter();
     2262        LogFlowThisFunc(("Entered locks OK\n"));
    22392263
    22402264        CheckComRCThrowRC(rc);
  • trunk/src/VBox/Main/VirtualBoxImpl.cpp

    r24298 r24345  
    389389            throw rc;
    390390
     391
     392#ifdef DEBUG
     393        LogFlowThisFunc(("Dumping media backreferences\n"));
     394        dumpAllBackRefs();
     395#endif
     396
    391397        /* net services */
    392398        for (settings::DHCPServersList::const_iterator it = m->pMainConfigFile->llDhcpServers.begin();
     
    19081914// public methods only for internal purposes
    19091915/////////////////////////////////////////////////////////////////////////////
     1916
     1917#ifdef DEBUG
     1918void 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
    19101936
    19111937/**
     
    32743300 *
    32753301 * @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.
    32773303 *
    32783304 * When @a aSaveRegistry is @c true, this operation may fail because of the
  • trunk/src/VBox/Main/include/MediumImpl.h

    r24273 r24345  
    163163                       const Guid &aSnapshotId = Guid::Empty);
    164164
     165#ifdef DEBUG
     166    void dumpBackRefs();
     167#endif
     168
    165169    const Guid& id() const;
    166170    MediumState_T state() const;
    167171    const Utf8Str& location() const;
    168172    const Utf8Str& locationFull() const;
     173    uint64_t size() const;
    169174
    170175    const Guid* getFirstMachineBackrefId() const;
     
    244249
    245250    HRESULT prepareDiscard(MergeChain * &aChain);
    246     HRESULT discard(ComObjPtr<Progress> &aProgress, MergeChain *aChain);
     251    HRESULT discard(ComObjPtr<Progress> &aProgress, ULONG ulWeight, MergeChain *aChain);
    247252    void cancelDiscard(MergeChain *aChain);
    248253
  • trunk/src/VBox/Main/include/VirtualBoxImpl.h

    r24298 r24345  
    7878    friend class CallbackEvent;
    7979
    80     VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT (VirtualBox)
     80    VIRTUALBOXBASE_ADD_ERRORINFO_SUPPORT(VirtualBox)
    8181
    8282    DECLARE_CLASSFACTORY_SINGLETON(VirtualBox)
     
    179179
    180180    /* public methods only for internal purposes */
     181#ifdef DEBUG
     182    void dumpAllBackRefs();
     183#endif
    181184
    182185    HRESULT postEvent(Event *event);
Note: See TracChangeset for help on using the changeset viewer.

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