VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/MachineImplCloneVM.cpp@ 57019

Last change on this file since 57019 was 56577, checked in by vboxsync, 10 years ago

Main/MachineImplCloneVM.cpp: fix cloning of VMs with floppy (handled regularly, like writethrough disks) and DVD attachments, which previously could lead to hangs (especially when the image was located in the VM directory), and some tweaking of the error handling code path to make it more robust.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 64.2 KB
Line 
1/* $Id: MachineImplCloneVM.cpp 56577 2015-06-22 15:08:04Z vboxsync $ */
2/** @file
3 * Implementation of MachineCloneVM
4 */
5
6/*
7 * Copyright (C) 2011-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#include "MachineImplCloneVM.h"
19
20#include "VirtualBoxImpl.h"
21#include "MediumImpl.h"
22#include "HostImpl.h"
23
24#include <iprt/path.h>
25#include <iprt/dir.h>
26#include <iprt/cpp/utils.h>
27#ifdef DEBUG_poetzsch
28# include <iprt/stream.h>
29#endif
30
31#include <VBox/com/list.h>
32#include <VBox/com/MultiResult.h>
33
34// typedefs
35/////////////////////////////////////////////////////////////////////////////
36
37typedef struct
38{
39 Utf8Str strBaseName;
40 ComPtr<IMedium> pMedium;
41 uint32_t uIdx;
42 ULONG uWeight;
43} MEDIUMTASK;
44
45typedef struct
46{
47 RTCList<MEDIUMTASK> chain;
48 DeviceType_T devType;
49 bool fCreateDiffs;
50 bool fAttachLinked;
51} MEDIUMTASKCHAIN;
52
53typedef struct
54{
55 Guid snapshotUuid;
56 Utf8Str strSaveStateFile;
57 ULONG uWeight;
58} SAVESTATETASK;
59
60// The private class
61/////////////////////////////////////////////////////////////////////////////
62
63struct MachineCloneVMPrivate
64{
65 MachineCloneVMPrivate(MachineCloneVM *a_q, ComObjPtr<Machine> &a_pSrcMachine, ComObjPtr<Machine> &a_pTrgMachine,
66 CloneMode_T a_mode, const RTCList<CloneOptions_T> &opts)
67 : q_ptr(a_q)
68 , p(a_pSrcMachine)
69 , pSrcMachine(a_pSrcMachine)
70 , pTrgMachine(a_pTrgMachine)
71 , mode(a_mode)
72 , options(opts)
73 {}
74
75 /* Thread management */
76 int startWorker()
77 {
78 return RTThreadCreate(NULL,
79 MachineCloneVMPrivate::workerThread,
80 static_cast<void*>(this),
81 0,
82 RTTHREADTYPE_MAIN_WORKER,
83 0,
84 "MachineClone");
85 }
86
87 static int workerThread(RTTHREAD /* Thread */, void *pvUser)
88 {
89 MachineCloneVMPrivate *pTask = static_cast<MachineCloneVMPrivate*>(pvUser);
90 AssertReturn(pTask, VERR_INVALID_POINTER);
91
92 HRESULT rc = pTask->q_ptr->run();
93
94 pTask->pProgress->i_notifyComplete(rc);
95
96 pTask->q_ptr->destroy();
97
98 return VINF_SUCCESS;
99 }
100
101 /* Private helper methods */
102
103 /* MachineCloneVM::start helper: */
104 HRESULT createMachineList(const ComPtr<ISnapshot> &pSnapshot, RTCList< ComObjPtr<Machine> > &machineList) const;
105 inline void updateProgressStats(MEDIUMTASKCHAIN &mtc, bool fAttachLinked, ULONG &uCount, ULONG &uTotalWeight) const;
106 inline HRESULT addSaveState(const ComObjPtr<Machine> &machine, bool fAttachCurrent, ULONG &uCount, ULONG &uTotalWeight);
107 inline HRESULT queryBaseName(const ComPtr<IMedium> &pMedium, Utf8Str &strBaseName) const;
108 HRESULT queryMediasForMachineState(const RTCList<ComObjPtr<Machine> > &machineList,
109 bool fAttachLinked, ULONG &uCount, ULONG &uTotalWeight);
110 HRESULT queryMediasForMachineAndChildStates(const RTCList<ComObjPtr<Machine> > &machineList,
111 bool fAttachLinked, ULONG &uCount, ULONG &uTotalWeight);
112 HRESULT queryMediasForAllStates(const RTCList<ComObjPtr<Machine> > &machineList, bool fAttachLinked, ULONG &uCount,
113 ULONG &uTotalWeight);
114
115 /* MachineCloneVM::run helper: */
116 bool findSnapshot(const settings::SnapshotsList &snl, const Guid &id, settings::Snapshot &sn) const;
117 void updateMACAddresses(settings::NetworkAdaptersList &nwl) const;
118 void updateMACAddresses(settings::SnapshotsList &sl) const;
119 void updateStorageLists(settings::StorageControllersList &sc, const Bstr &bstrOldId, const Bstr &bstrNewId) const;
120 void updateSnapshotStorageLists(settings::SnapshotsList &sl, const Bstr &bstrOldId, const Bstr &bstrNewId) const;
121 void updateStateFile(settings::SnapshotsList &snl, const Guid &id, const Utf8Str &strFile) const;
122 HRESULT createDifferencingMedium(const ComObjPtr<Machine> &pMachine, const ComObjPtr<Medium> &pParent,
123 const Utf8Str &strSnapshotFolder, RTCList<ComObjPtr<Medium> > &newMedia,
124 ComObjPtr<Medium> *ppDiff) const;
125 static int copyStateFileProgress(unsigned uPercentage, void *pvUser);
126
127 /* Private q and parent pointer */
128 MachineCloneVM *q_ptr;
129 ComObjPtr<Machine> p;
130
131 /* Private helper members */
132 ComObjPtr<Machine> pSrcMachine;
133 ComObjPtr<Machine> pTrgMachine;
134 ComPtr<IMachine> pOldMachineState;
135 ComObjPtr<Progress> pProgress;
136 Guid snapshotId;
137 CloneMode_T mode;
138 RTCList<CloneOptions_T> options;
139 RTCList<MEDIUMTASKCHAIN> llMedias;
140 RTCList<SAVESTATETASK> llSaveStateFiles; /* Snapshot UUID -> File path */
141};
142
143HRESULT MachineCloneVMPrivate::createMachineList(const ComPtr<ISnapshot> &pSnapshot,
144 RTCList< ComObjPtr<Machine> > &machineList) const
145{
146 HRESULT rc = S_OK;
147 Bstr name;
148 rc = pSnapshot->COMGETTER(Name)(name.asOutParam());
149 if (FAILED(rc)) return rc;
150
151 ComPtr<IMachine> pMachine;
152 rc = pSnapshot->COMGETTER(Machine)(pMachine.asOutParam());
153 if (FAILED(rc)) return rc;
154 machineList.append((Machine*)(IMachine*)pMachine);
155
156 SafeIfaceArray<ISnapshot> sfaChilds;
157 rc = pSnapshot->COMGETTER(Children)(ComSafeArrayAsOutParam(sfaChilds));
158 if (FAILED(rc)) return rc;
159 for (size_t i = 0; i < sfaChilds.size(); ++i)
160 {
161 rc = createMachineList(sfaChilds[i], machineList);
162 if (FAILED(rc)) return rc;
163 }
164
165 return rc;
166}
167
168void MachineCloneVMPrivate::updateProgressStats(MEDIUMTASKCHAIN &mtc, bool fAttachLinked,
169 ULONG &uCount, ULONG &uTotalWeight) const
170{
171 if (fAttachLinked)
172 {
173 /* Implicit diff creation as part of attach is a pretty cheap
174 * operation, and does only need one operation per attachment. */
175 ++uCount;
176 uTotalWeight += 1; /* 1MB per attachment */
177 }
178 else
179 {
180 /* Currently the copying of diff images involves reading at least
181 * the biggest parent in the previous chain. So even if the new
182 * diff image is small in size, it could need some time to create
183 * it. Adding the biggest size in the chain should balance this a
184 * little bit more, i.e. the weight is the sum of the data which
185 * needs to be read and written. */
186 ULONG uMaxWeight = 0;
187 for (size_t e = mtc.chain.size(); e > 0; --e)
188 {
189 MEDIUMTASK &mt = mtc.chain.at(e - 1);
190 mt.uWeight += uMaxWeight;
191
192 /* Calculate progress data */
193 ++uCount;
194 uTotalWeight += mt.uWeight;
195
196 /* Save the max size for better weighting of diff image
197 * creation. */
198 uMaxWeight = RT_MAX(uMaxWeight, mt.uWeight);
199 }
200 }
201}
202
203HRESULT MachineCloneVMPrivate::addSaveState(const ComObjPtr<Machine> &machine, bool fAttachCurrent, ULONG &uCount, ULONG &uTotalWeight)
204{
205 Bstr bstrSrcSaveStatePath;
206 HRESULT rc = machine->COMGETTER(StateFilePath)(bstrSrcSaveStatePath.asOutParam());
207 if (FAILED(rc)) return rc;
208 if (!bstrSrcSaveStatePath.isEmpty())
209 {
210 SAVESTATETASK sst;
211 if (fAttachCurrent)
212 {
213 /* Make this saved state part of "current state" of the target
214 * machine, whether it is part of a snapshot or not. */
215 sst.snapshotUuid.clear();
216 }
217 else
218 sst.snapshotUuid = machine->i_getSnapshotId();
219 sst.strSaveStateFile = bstrSrcSaveStatePath;
220 uint64_t cbSize;
221 int vrc = RTFileQuerySize(sst.strSaveStateFile.c_str(), &cbSize);
222 if (RT_FAILURE(vrc))
223 return p->setError(VBOX_E_IPRT_ERROR, p->tr("Could not query file size of '%s' (%Rrc)"),
224 sst.strSaveStateFile.c_str(), vrc);
225 /* same rule as above: count both the data which needs to
226 * be read and written */
227 sst.uWeight = (ULONG)(2 * (cbSize + _1M - 1) / _1M);
228 llSaveStateFiles.append(sst);
229 ++uCount;
230 uTotalWeight += sst.uWeight;
231 }
232 return S_OK;
233}
234
235HRESULT MachineCloneVMPrivate::queryBaseName(const ComPtr<IMedium> &pMedium, Utf8Str &strBaseName) const
236{
237 ComPtr<IMedium> pBaseMedium;
238 HRESULT rc = pMedium->COMGETTER(Base)(pBaseMedium.asOutParam());
239 if (FAILED(rc)) return rc;
240 Bstr bstrBaseName;
241 rc = pBaseMedium->COMGETTER(Name)(bstrBaseName.asOutParam());
242 if (FAILED(rc)) return rc;
243 strBaseName = bstrBaseName;
244 return rc;
245}
246
247HRESULT MachineCloneVMPrivate::queryMediasForMachineState(const RTCList<ComObjPtr<Machine> > &machineList,
248 bool fAttachLinked, ULONG &uCount, ULONG &uTotalWeight)
249{
250 /* This mode is pretty straightforward. We didn't need to know about any
251 * parent/children relationship and therefore simply adding all directly
252 * attached images of the source VM as cloning targets. The IMedium code
253 * take than care to merge any (possibly) existing parents into the new
254 * image. */
255 HRESULT rc = S_OK;
256 for (size_t i = 0; i < machineList.size(); ++i)
257 {
258 const ComObjPtr<Machine> &machine = machineList.at(i);
259 /* If this is the Snapshot Machine we want to clone, we need to
260 * create a new diff file for the new "current state". */
261 const bool fCreateDiffs = (machine == pOldMachineState);
262 /* Add all attachments of the different machines to a worker list. */
263 SafeIfaceArray<IMediumAttachment> sfaAttachments;
264 rc = machine->COMGETTER(MediumAttachments)(ComSafeArrayAsOutParam(sfaAttachments));
265 if (FAILED(rc)) return rc;
266 for (size_t a = 0; a < sfaAttachments.size(); ++a)
267 {
268 const ComPtr<IMediumAttachment> &pAtt = sfaAttachments[a];
269 DeviceType_T type;
270 rc = pAtt->COMGETTER(Type)(&type);
271 if (FAILED(rc)) return rc;
272
273 /* Only harddisks and floppies are of interest. */
274 if ( type != DeviceType_HardDisk
275 && type != DeviceType_Floppy)
276 continue;
277
278 /* Valid medium attached? */
279 ComPtr<IMedium> pSrcMedium;
280 rc = pAtt->COMGETTER(Medium)(pSrcMedium.asOutParam());
281 if (FAILED(rc)) return rc;
282
283 if (pSrcMedium.isNull())
284 continue;
285
286 /* Create the medium task chain. In this case it will always
287 * contain one image only. */
288 MEDIUMTASKCHAIN mtc;
289 mtc.devType = type;
290 mtc.fCreateDiffs = fCreateDiffs;
291 mtc.fAttachLinked = fAttachLinked;
292
293 /* Refresh the state so that the file size get read. */
294 MediumState_T e;
295 rc = pSrcMedium->RefreshState(&e);
296 if (FAILED(rc)) return rc;
297 LONG64 lSize;
298 rc = pSrcMedium->COMGETTER(Size)(&lSize);
299 if (FAILED(rc)) return rc;
300
301 MEDIUMTASK mt;
302 mt.uIdx = UINT32_MAX; /* No read/write optimization possible. */
303
304 /* Save the base name. */
305 rc = queryBaseName(pSrcMedium, mt.strBaseName);
306 if (FAILED(rc)) return rc;
307
308 /* Save the current medium, for later cloning. */
309 mt.pMedium = pSrcMedium;
310 if (fAttachLinked)
311 mt.uWeight = 0; /* dummy */
312 else
313 mt.uWeight = (ULONG)((lSize + _1M - 1) / _1M);
314 mtc.chain.append(mt);
315
316 /* Update the progress info. */
317 updateProgressStats(mtc, fAttachLinked, uCount, uTotalWeight);
318 /* Append the list of images which have to be cloned. */
319 llMedias.append(mtc);
320 }
321 /* Add the save state files of this machine if there is one. */
322 rc = addSaveState(machine, true /*fAttachCurrent*/, uCount, uTotalWeight);
323 if (FAILED(rc)) return rc;
324 }
325
326 return rc;
327}
328
329HRESULT MachineCloneVMPrivate::queryMediasForMachineAndChildStates(const RTCList<ComObjPtr<Machine> > &machineList,
330 bool fAttachLinked, ULONG &uCount, ULONG &uTotalWeight)
331{
332 /* This is basically a three step approach. First select all medias
333 * directly or indirectly involved in the clone. Second create a histogram
334 * of the usage of all that medias. Third select the medias which are
335 * directly attached or have more than one directly/indirectly used child
336 * in the new clone. Step one and two are done in the first loop.
337 *
338 * Example of the histogram counts after going through 3 attachments from
339 * bottom to top:
340 *
341 * 3
342 * |
343 * -> 3
344 * / \
345 * 2 1 <-
346 * /
347 * -> 2
348 * / \
349 * -> 1 1
350 * \
351 * 1 <-
352 *
353 * Whenever the histogram count is changing compared to the previous one we
354 * need to include that image in the cloning step (Marked with <-). If we
355 * start at zero even the directly attached images are automatically
356 * included.
357 *
358 * Note: This still leads to media chains which can have the same medium
359 * included. This case is handled in "run" and therefore not critical, but
360 * it leads to wrong progress infos which isn't nice. */
361
362 Assert(!fAttachLinked);
363 HRESULT rc = S_OK;
364 std::map<ComPtr<IMedium>, uint32_t> mediaHist; /* Our usage histogram for the medias */
365 for (size_t i = 0; i < machineList.size(); ++i)
366 {
367 const ComObjPtr<Machine> &machine = machineList.at(i);
368 /* If this is the Snapshot Machine we want to clone, we need to
369 * create a new diff file for the new "current state". */
370 const bool fCreateDiffs = (machine == pOldMachineState);
371 /* Add all attachments (and their parents) of the different
372 * machines to a worker list. */
373 SafeIfaceArray<IMediumAttachment> sfaAttachments;
374 rc = machine->COMGETTER(MediumAttachments)(ComSafeArrayAsOutParam(sfaAttachments));
375 if (FAILED(rc)) return rc;
376 for (size_t a = 0; a < sfaAttachments.size(); ++a)
377 {
378 const ComPtr<IMediumAttachment> &pAtt = sfaAttachments[a];
379 DeviceType_T type;
380 rc = pAtt->COMGETTER(Type)(&type);
381 if (FAILED(rc)) return rc;
382
383 /* Only harddisks and floppies are of interest. */
384 if ( type != DeviceType_HardDisk
385 && type != DeviceType_Floppy)
386 continue;
387
388 /* Valid medium attached? */
389 ComPtr<IMedium> pSrcMedium;
390 rc = pAtt->COMGETTER(Medium)(pSrcMedium.asOutParam());
391 if (FAILED(rc)) return rc;
392
393 if (pSrcMedium.isNull())
394 continue;
395
396 MEDIUMTASKCHAIN mtc;
397 mtc.devType = type;
398 mtc.fCreateDiffs = fCreateDiffs;
399 mtc.fAttachLinked = fAttachLinked;
400
401 while (!pSrcMedium.isNull())
402 {
403 /* Build a histogram of used medias and the parent chain. */
404 ++mediaHist[pSrcMedium];
405
406 /* Refresh the state so that the file size get read. */
407 MediumState_T e;
408 rc = pSrcMedium->RefreshState(&e);
409 if (FAILED(rc)) return rc;
410 LONG64 lSize;
411 rc = pSrcMedium->COMGETTER(Size)(&lSize);
412 if (FAILED(rc)) return rc;
413
414 MEDIUMTASK mt;
415 mt.uIdx = UINT32_MAX;
416 mt.pMedium = pSrcMedium;
417 mt.uWeight = (ULONG)((lSize + _1M - 1) / _1M);
418 mtc.chain.append(mt);
419
420 /* Query next parent. */
421 rc = pSrcMedium->COMGETTER(Parent)(pSrcMedium.asOutParam());
422 if (FAILED(rc)) return rc;
423 }
424
425 llMedias.append(mtc);
426 }
427 /* Add the save state files of this machine if there is one. */
428 rc = addSaveState(machine, false /*fAttachCurrent*/, uCount, uTotalWeight);
429 if (FAILED(rc)) return rc;
430 /* If this is the newly created current state, make sure that the
431 * saved state is also attached to it. */
432 if (fCreateDiffs)
433 {
434 rc = addSaveState(machine, true /*fAttachCurrent*/, uCount, uTotalWeight);
435 if (FAILED(rc)) return rc;
436 }
437 }
438 /* Build up the index list of the image chain. Unfortunately we can't do
439 * that in the previous loop, cause there we go from child -> parent and
440 * didn't know how many are between. */
441 for (size_t i = 0; i < llMedias.size(); ++i)
442 {
443 uint32_t uIdx = 0;
444 MEDIUMTASKCHAIN &mtc = llMedias.at(i);
445 for (size_t a = mtc.chain.size(); a > 0; --a)
446 mtc.chain[a - 1].uIdx = uIdx++;
447 }
448#ifdef DEBUG_poetzsch
449 /* Print the histogram */
450 std::map<ComPtr<IMedium>, uint32_t>::iterator it;
451 for (it = mediaHist.begin(); it != mediaHist.end(); ++it)
452 {
453 Bstr bstrSrcName;
454 rc = (*it).first->COMGETTER(Name)(bstrSrcName.asOutParam());
455 if (FAILED(rc)) return rc;
456 RTPrintf("%ls: %d\n", bstrSrcName.raw(), (*it).second);
457 }
458#endif
459 /* Go over every medium in the list and check if it either a directly
460 * attached disk or has more than one children. If so it needs to be
461 * replicated. Also we have to make sure that any direct or indirect
462 * children knows of the new parent (which doesn't necessarily mean it
463 * is a direct children in the source chain). */
464 for (size_t i = 0; i < llMedias.size(); ++i)
465 {
466 MEDIUMTASKCHAIN &mtc = llMedias.at(i);
467 RTCList<MEDIUMTASK> newChain;
468 uint32_t used = 0;
469 for (size_t a = 0; a < mtc.chain.size(); ++a)
470 {
471 const MEDIUMTASK &mt = mtc.chain.at(a);
472 uint32_t hist = mediaHist[mt.pMedium];
473#ifdef DEBUG_poetzsch
474 Bstr bstrSrcName;
475 rc = mt.pMedium->COMGETTER(Name)(bstrSrcName.asOutParam());
476 if (FAILED(rc)) return rc;
477 RTPrintf("%ls: %d (%d)\n", bstrSrcName.raw(), hist, used);
478#endif
479 /* Check if there is a "step" in the histogram when going the chain
480 * upwards. If so, we need this image, cause there is another branch
481 * from here in the cloned VM. */
482 if (hist > used)
483 {
484 newChain.append(mt);
485 used = hist;
486 }
487 }
488 /* Make sure we always using the old base name as new base name, even
489 * if the base is a differencing image in the source VM (with the UUID
490 * as name). */
491 rc = queryBaseName(newChain.last().pMedium, newChain.last().strBaseName);
492 if (FAILED(rc)) return rc;
493 /* Update the old medium chain with the updated one. */
494 mtc.chain = newChain;
495 /* Update the progress info. */
496 updateProgressStats(mtc, fAttachLinked, uCount, uTotalWeight);
497 }
498
499 return rc;
500}
501
502HRESULT MachineCloneVMPrivate::queryMediasForAllStates(const RTCList<ComObjPtr<Machine> > &machineList,
503 bool fAttachLinked, ULONG &uCount, ULONG &uTotalWeight)
504{
505 /* In this case we create a exact copy of the original VM. This means just
506 * adding all directly and indirectly attached disk images to the worker
507 * list. */
508 Assert(!fAttachLinked);
509 HRESULT rc = S_OK;
510 for (size_t i = 0; i < machineList.size(); ++i)
511 {
512 const ComObjPtr<Machine> &machine = machineList.at(i);
513 /* If this is the Snapshot Machine we want to clone, we need to
514 * create a new diff file for the new "current state". */
515 const bool fCreateDiffs = (machine == pOldMachineState);
516 /* Add all attachments (and their parents) of the different
517 * machines to a worker list. */
518 SafeIfaceArray<IMediumAttachment> sfaAttachments;
519 rc = machine->COMGETTER(MediumAttachments)(ComSafeArrayAsOutParam(sfaAttachments));
520 if (FAILED(rc)) return rc;
521 for (size_t a = 0; a < sfaAttachments.size(); ++a)
522 {
523 const ComPtr<IMediumAttachment> &pAtt = sfaAttachments[a];
524 DeviceType_T type;
525 rc = pAtt->COMGETTER(Type)(&type);
526 if (FAILED(rc)) return rc;
527
528 /* Only harddisks and floppies are of interest. */
529 if ( type != DeviceType_HardDisk
530 && type != DeviceType_Floppy)
531 continue;
532
533 /* Valid medium attached? */
534 ComPtr<IMedium> pSrcMedium;
535 rc = pAtt->COMGETTER(Medium)(pSrcMedium.asOutParam());
536 if (FAILED(rc)) return rc;
537
538 if (pSrcMedium.isNull())
539 continue;
540
541 /* Build up a child->parent list of this attachment. (Note: we are
542 * not interested of any child's not attached to this VM. So this
543 * will not create a full copy of the base/child relationship.) */
544 MEDIUMTASKCHAIN mtc;
545 mtc.devType = type;
546 mtc.fCreateDiffs = fCreateDiffs;
547 mtc.fAttachLinked = fAttachLinked;
548
549 while (!pSrcMedium.isNull())
550 {
551 /* Refresh the state so that the file size get read. */
552 MediumState_T e;
553 rc = pSrcMedium->RefreshState(&e);
554 if (FAILED(rc)) return rc;
555 LONG64 lSize;
556 rc = pSrcMedium->COMGETTER(Size)(&lSize);
557 if (FAILED(rc)) return rc;
558
559 /* Save the current medium, for later cloning. */
560 MEDIUMTASK mt;
561 mt.uIdx = UINT32_MAX;
562 mt.pMedium = pSrcMedium;
563 mt.uWeight = (ULONG)((lSize + _1M - 1) / _1M);
564 mtc.chain.append(mt);
565
566 /* Query next parent. */
567 rc = pSrcMedium->COMGETTER(Parent)(pSrcMedium.asOutParam());
568 if (FAILED(rc)) return rc;
569 }
570 /* Update the progress info. */
571 updateProgressStats(mtc, fAttachLinked, uCount, uTotalWeight);
572 /* Append the list of images which have to be cloned. */
573 llMedias.append(mtc);
574 }
575 /* Add the save state files of this machine if there is one. */
576 rc = addSaveState(machine, false /*fAttachCurrent*/, uCount, uTotalWeight);
577 if (FAILED(rc)) return rc;
578 /* If this is the newly created current state, make sure that the
579 * saved state is also attached to it. */
580 if (fCreateDiffs)
581 {
582 rc = addSaveState(machine, true /*fAttachCurrent*/, uCount, uTotalWeight);
583 if (FAILED(rc)) return rc;
584 }
585 }
586 /* Build up the index list of the image chain. Unfortunately we can't do
587 * that in the previous loop, cause there we go from child -> parent and
588 * didn't know how many are between. */
589 for (size_t i = 0; i < llMedias.size(); ++i)
590 {
591 uint32_t uIdx = 0;
592 MEDIUMTASKCHAIN &mtc = llMedias.at(i);
593 for (size_t a = mtc.chain.size(); a > 0; --a)
594 mtc.chain[a - 1].uIdx = uIdx++;
595 }
596
597 return rc;
598}
599
600bool MachineCloneVMPrivate::findSnapshot(const settings::SnapshotsList &snl, const Guid &id, settings::Snapshot &sn) const
601{
602 settings::SnapshotsList::const_iterator it;
603 for (it = snl.begin(); it != snl.end(); ++it)
604 {
605 if (it->uuid == id)
606 {
607 sn = (*it);
608 return true;
609 }
610 else if (!it->llChildSnapshots.empty())
611 {
612 if (findSnapshot(it->llChildSnapshots, id, sn))
613 return true;
614 }
615 }
616 return false;
617}
618
619void MachineCloneVMPrivate::updateMACAddresses(settings::NetworkAdaptersList &nwl) const
620{
621 const bool fNotNAT = options.contains(CloneOptions_KeepNATMACs);
622 settings::NetworkAdaptersList::iterator it;
623 for (it = nwl.begin(); it != nwl.end(); ++it)
624 {
625 if ( fNotNAT
626 && it->mode == NetworkAttachmentType_NAT)
627 continue;
628 Host::i_generateMACAddress(it->strMACAddress);
629 }
630}
631
632void MachineCloneVMPrivate::updateMACAddresses(settings::SnapshotsList &sl) const
633{
634 settings::SnapshotsList::iterator it;
635 for (it = sl.begin(); it != sl.end(); ++it)
636 {
637 updateMACAddresses(it->hardware.llNetworkAdapters);
638 if (!it->llChildSnapshots.empty())
639 updateMACAddresses(it->llChildSnapshots);
640 }
641}
642
643void MachineCloneVMPrivate::updateStorageLists(settings::StorageControllersList &sc,
644 const Bstr &bstrOldId, const Bstr &bstrNewId) const
645{
646 settings::StorageControllersList::iterator it3;
647 for (it3 = sc.begin();
648 it3 != sc.end();
649 ++it3)
650 {
651 settings::AttachedDevicesList &llAttachments = it3->llAttachedDevices;
652 settings::AttachedDevicesList::iterator it4;
653 for (it4 = llAttachments.begin();
654 it4 != llAttachments.end();
655 ++it4)
656 {
657 if ( ( it4->deviceType == DeviceType_HardDisk
658 || it4->deviceType == DeviceType_Floppy)
659 && it4->uuid == bstrOldId)
660 {
661 it4->uuid = bstrNewId;
662 }
663 }
664 }
665}
666
667void MachineCloneVMPrivate::updateSnapshotStorageLists(settings::SnapshotsList &sl, const Bstr &bstrOldId,
668 const Bstr &bstrNewId) const
669{
670 settings::SnapshotsList::iterator it;
671 for ( it = sl.begin();
672 it != sl.end();
673 ++it)
674 {
675 updateStorageLists(it->storage.llStorageControllers, bstrOldId, bstrNewId);
676 if (!it->llChildSnapshots.empty())
677 updateSnapshotStorageLists(it->llChildSnapshots, bstrOldId, bstrNewId);
678 }
679}
680
681void MachineCloneVMPrivate::updateStateFile(settings::SnapshotsList &snl, const Guid &id, const Utf8Str &strFile) const
682{
683 settings::SnapshotsList::iterator it;
684 for (it = snl.begin(); it != snl.end(); ++it)
685 {
686 if (it->uuid == id)
687 it->strStateFile = strFile;
688 else if (!it->llChildSnapshots.empty())
689 updateStateFile(it->llChildSnapshots, id, strFile);
690 }
691}
692
693HRESULT MachineCloneVMPrivate::createDifferencingMedium(const ComObjPtr<Machine> &pMachine, const ComObjPtr<Medium> &pParent,
694 const Utf8Str &strSnapshotFolder, RTCList<ComObjPtr<Medium> > &newMedia,
695 ComObjPtr<Medium> *ppDiff) const
696{
697 HRESULT rc = S_OK;
698 try
699 {
700 // check validity of parent object
701 {
702 AutoReadLock alock(pParent COMMA_LOCKVAL_SRC_POS);
703 Bstr bstrSrcId;
704 rc = pParent->COMGETTER(Id)(bstrSrcId.asOutParam());
705 if (FAILED(rc)) throw rc;
706 }
707 ComObjPtr<Medium> diff;
708 diff.createObject();
709 rc = diff->init(p->i_getVirtualBox(),
710 pParent->i_getPreferredDiffFormat(),
711 Utf8StrFmt("%s%c", strSnapshotFolder.c_str(), RTPATH_DELIMITER),
712 Guid::Empty /* empty media registry */,
713 DeviceType_HardDisk);
714 if (FAILED(rc)) throw rc;
715
716 MediumLockList *pMediumLockList(new MediumLockList());
717 rc = diff->i_createMediumLockList(true /* fFailIfInaccessible */,
718 true /* fMediumLockWrite */,
719 false /* fMediumLockWriteAll */,
720 pParent,
721 *pMediumLockList);
722 if (FAILED(rc)) throw rc;
723 rc = pMediumLockList->Lock();
724 if (FAILED(rc)) throw rc;
725
726 /* this already registers the new diff image */
727 rc = pParent->i_createDiffStorage(diff, MediumVariant_Standard,
728 pMediumLockList,
729 NULL /* aProgress */,
730 true /* aWait */);
731 delete pMediumLockList;
732 if (FAILED(rc)) throw rc;
733 /* Remember created medium. */
734 newMedia.append(diff);
735 *ppDiff = diff;
736 }
737 catch (HRESULT rc2)
738 {
739 rc = rc2;
740 }
741 catch (...)
742 {
743 rc = VirtualBoxBase::handleUnexpectedExceptions(pMachine, RT_SRC_POS);
744 }
745
746 return rc;
747}
748
749/* static */
750int MachineCloneVMPrivate::copyStateFileProgress(unsigned uPercentage, void *pvUser)
751{
752 ComObjPtr<Progress> pProgress = *static_cast< ComObjPtr<Progress>* >(pvUser);
753
754 BOOL fCanceled = false;
755 HRESULT rc = pProgress->COMGETTER(Canceled)(&fCanceled);
756 if (FAILED(rc)) return VERR_GENERAL_FAILURE;
757 /* If canceled by the user tell it to the copy operation. */
758 if (fCanceled) return VERR_CANCELLED;
759 /* Set the new process. */
760 rc = pProgress->SetCurrentOperationProgress(uPercentage);
761 if (FAILED(rc)) return VERR_GENERAL_FAILURE;
762
763 return VINF_SUCCESS;
764}
765
766// The public class
767/////////////////////////////////////////////////////////////////////////////
768
769MachineCloneVM::MachineCloneVM(ComObjPtr<Machine> pSrcMachine, ComObjPtr<Machine> pTrgMachine, CloneMode_T mode,
770 const RTCList<CloneOptions_T> &opts) :
771 d_ptr(new MachineCloneVMPrivate(this, pSrcMachine, pTrgMachine, mode, opts))
772{
773}
774
775MachineCloneVM::~MachineCloneVM()
776{
777 delete d_ptr;
778}
779
780HRESULT MachineCloneVM::start(IProgress **pProgress)
781{
782 DPTR(MachineCloneVM);
783 ComObjPtr<Machine> &p = d->p;
784
785 HRESULT rc;
786 try
787 {
788 /** @todo r=klaus this code cannot deal with someone crazy specifying
789 * IMachine corresponding to a mutable machine as d->pSrcMachine */
790 if (d->pSrcMachine->i_isSessionMachine())
791 throw p->setError(E_INVALIDARG, "The source machine is mutable");
792
793 /* Handle the special case that someone is requesting a _full_ clone
794 * with all snapshots (and the current state), but uses a snapshot
795 * machine (and not the current one) as source machine. In this case we
796 * just replace the source (snapshot) machine with the current machine. */
797 if ( d->mode == CloneMode_AllStates
798 && d->pSrcMachine->i_isSnapshotMachine())
799 {
800 Bstr bstrSrcMachineId;
801 rc = d->pSrcMachine->COMGETTER(Id)(bstrSrcMachineId.asOutParam());
802 if (FAILED(rc)) throw rc;
803 ComPtr<IMachine> newSrcMachine;
804 rc = d->pSrcMachine->i_getVirtualBox()->FindMachine(bstrSrcMachineId.raw(), newSrcMachine.asOutParam());
805 if (FAILED(rc)) throw rc;
806 d->pSrcMachine = (Machine*)(IMachine*)newSrcMachine;
807 }
808 bool fSubtreeIncludesCurrent = false;
809 ComObjPtr<Machine> pCurrState;
810 if (d->mode == CloneMode_MachineAndChildStates)
811 {
812 if (d->pSrcMachine->i_isSnapshotMachine())
813 {
814 /* find machine object for current snapshot of current state */
815 Bstr bstrSrcMachineId;
816 rc = d->pSrcMachine->COMGETTER(Id)(bstrSrcMachineId.asOutParam());
817 if (FAILED(rc)) throw rc;
818 ComPtr<IMachine> pCurr;
819 rc = d->pSrcMachine->i_getVirtualBox()->FindMachine(bstrSrcMachineId.raw(), pCurr.asOutParam());
820 if (FAILED(rc)) throw rc;
821 if (pCurr.isNull())
822 throw p->setError(VBOX_E_OBJECT_NOT_FOUND);
823 pCurrState = (Machine *)(IMachine *)pCurr;
824 ComPtr<ISnapshot> pSnapshot;
825 rc = pCurrState->COMGETTER(CurrentSnapshot)(pSnapshot.asOutParam());
826 if (FAILED(rc)) throw rc;
827 if (pSnapshot.isNull())
828 throw p->setError(VBOX_E_OBJECT_NOT_FOUND);
829 ComPtr<IMachine> pCurrSnapMachine;
830 rc = pSnapshot->COMGETTER(Machine)(pCurrSnapMachine.asOutParam());
831 if (FAILED(rc)) throw rc;
832 if (pCurrSnapMachine.isNull())
833 throw p->setError(VBOX_E_OBJECT_NOT_FOUND);
834
835 /* now check if there is a parent chain which leads to the
836 * snapshot machine defining the subtree. */
837 while (!pSnapshot.isNull())
838 {
839 ComPtr<IMachine> pSnapMachine;
840 rc = pSnapshot->COMGETTER(Machine)(pSnapMachine.asOutParam());
841 if (FAILED(rc)) throw rc;
842 if (pSnapMachine.isNull())
843 throw p->setError(VBOX_E_OBJECT_NOT_FOUND);
844 if (pSnapMachine == d->pSrcMachine)
845 {
846 fSubtreeIncludesCurrent = true;
847 break;
848 }
849 rc = pSnapshot->COMGETTER(Parent)(pSnapshot.asOutParam());
850 if (FAILED(rc)) throw rc;
851 }
852 }
853 else
854 {
855 /* If the subtree is only the Current State simply use the
856 * 'machine' case for cloning. It is easier to understand. */
857 d->mode = CloneMode_MachineState;
858 }
859 }
860
861 /* Lock the target machine early (so nobody mess around with it in the meantime). */
862 AutoWriteLock trgLock(d->pTrgMachine COMMA_LOCKVAL_SRC_POS);
863
864 if (d->pSrcMachine->i_isSnapshotMachine())
865 d->snapshotId = d->pSrcMachine->i_getSnapshotId();
866
867 /* Add the current machine and all snapshot machines below this machine
868 * in a list for further processing. */
869 RTCList< ComObjPtr<Machine> > machineList;
870
871 /* Include current state? */
872 if ( d->mode == CloneMode_MachineState
873 || d->mode == CloneMode_AllStates)
874 machineList.append(d->pSrcMachine);
875 /* Should be done a depth copy with all child snapshots? */
876 if ( d->mode == CloneMode_MachineAndChildStates
877 || d->mode == CloneMode_AllStates)
878 {
879 ULONG cSnapshots = 0;
880 rc = d->pSrcMachine->COMGETTER(SnapshotCount)(&cSnapshots);
881 if (FAILED(rc)) throw rc;
882 if (cSnapshots > 0)
883 {
884 Utf8Str id;
885 if (d->mode == CloneMode_MachineAndChildStates)
886 id = d->snapshotId.toString();
887 ComPtr<ISnapshot> pSnapshot;
888 rc = d->pSrcMachine->FindSnapshot(Bstr(id).raw(), pSnapshot.asOutParam());
889 if (FAILED(rc)) throw rc;
890 rc = d->createMachineList(pSnapshot, machineList);
891 if (FAILED(rc)) throw rc;
892 if (d->mode == CloneMode_MachineAndChildStates)
893 {
894 if (fSubtreeIncludesCurrent)
895 {
896 if (pCurrState.isNull())
897 throw p->setError(VBOX_E_OBJECT_NOT_FOUND);
898 machineList.append(pCurrState);
899 }
900 else
901 {
902 rc = pSnapshot->COMGETTER(Machine)(d->pOldMachineState.asOutParam());
903 if (FAILED(rc)) throw rc;
904 }
905 }
906 }
907 }
908
909 /* We have different approaches for getting the medias which needs to
910 * be replicated based on the clone mode the user requested (this is
911 * mostly about the full clone mode).
912 * MachineState:
913 * - Only the images which are directly attached to an source VM will
914 * be cloned. Any parent disks in the original chain will be merged
915 * into the final cloned disk.
916 * MachineAndChildStates:
917 * - In this case we search for images which have more than one
918 * children in the cloned VM or are directly attached to the new VM.
919 * All others will be merged into the remaining images which are
920 * cloned.
921 * This case is the most complicated one and needs several iterations
922 * to make sure we are only cloning images which are really
923 * necessary.
924 * AllStates:
925 * - All disks which are directly or indirectly attached to the
926 * original VM are cloned.
927 *
928 * Note: If you change something generic in one of the methods its
929 * likely that it need to be changed in the others as well! */
930 ULONG uCount = 2; /* One init task and the machine creation. */
931 ULONG uTotalWeight = 2; /* The init task and the machine creation is worth one. */
932 bool fAttachLinked = d->options.contains(CloneOptions_Link); /* Linked clones requested? */
933 switch (d->mode)
934 {
935 case CloneMode_MachineState: d->queryMediasForMachineState(machineList, fAttachLinked,
936 uCount, uTotalWeight);
937 break;
938 case CloneMode_MachineAndChildStates: d->queryMediasForMachineAndChildStates(machineList, fAttachLinked,
939 uCount, uTotalWeight);
940 break;
941 case CloneMode_AllStates: d->queryMediasForAllStates(machineList, fAttachLinked, uCount,
942 uTotalWeight);
943 break;
944 }
945
946 /* Now create the progress project, so the user knows whats going on. */
947 rc = d->pProgress.createObject();
948 if (FAILED(rc)) throw rc;
949 rc = d->pProgress->init(p->i_getVirtualBox(),
950 static_cast<IMachine*>(d->pSrcMachine) /* aInitiator */,
951 Bstr(p->tr("Cloning Machine")).raw(),
952 true /* fCancellable */,
953 uCount,
954 uTotalWeight,
955 Bstr(p->tr("Initialize Cloning")).raw(),
956 1);
957 if (FAILED(rc)) throw rc;
958
959 int vrc = d->startWorker();
960
961 if (RT_FAILURE(vrc))
962 p->setError(VBOX_E_IPRT_ERROR, "Could not create machine clone thread (%Rrc)", vrc);
963 }
964 catch (HRESULT rc2)
965 {
966 rc = rc2;
967 }
968
969 if (SUCCEEDED(rc))
970 d->pProgress.queryInterfaceTo(pProgress);
971
972 return rc;
973}
974
975HRESULT MachineCloneVM::run()
976{
977 DPTR(MachineCloneVM);
978 ComObjPtr<Machine> &p = d->p;
979
980 AutoCaller autoCaller(p);
981 if (FAILED(autoCaller.rc())) return autoCaller.rc();
982
983 AutoReadLock srcLock(p COMMA_LOCKVAL_SRC_POS);
984 AutoWriteLock trgLock(d->pTrgMachine COMMA_LOCKVAL_SRC_POS);
985
986 HRESULT rc = S_OK;
987
988 /*
989 * Todo:
990 * - What about log files?
991 */
992
993 /* Where should all the media go? */
994 Utf8Str strTrgSnapshotFolder;
995 Utf8Str strTrgMachineFolder = d->pTrgMachine->i_getSettingsFileFull();
996 strTrgMachineFolder.stripFilename();
997
998 RTCList<ComObjPtr<Medium> > newMedia; /* All created images */
999 RTCList<Utf8Str> newFiles; /* All extra created files (save states, ...) */
1000 try
1001 {
1002 /* Copy all the configuration from this machine to an empty
1003 * configuration dataset. */
1004 settings::MachineConfigFile trgMCF = *d->pSrcMachine->mData->pMachineConfigFile;
1005
1006 /* Reset media registry. */
1007 trgMCF.mediaRegistry.llHardDisks.clear();
1008 trgMCF.mediaRegistry.llDvdImages.clear();
1009 trgMCF.mediaRegistry.llFloppyImages.clear();
1010 /* If we got a valid snapshot id, replace the hardware/storage section
1011 * with the stuff from the snapshot. */
1012 settings::Snapshot sn;
1013
1014 if (d->snapshotId.isValid() && !d->snapshotId.isZero())
1015 if (!d->findSnapshot(trgMCF.llFirstSnapshot, d->snapshotId, sn))
1016 throw p->setError(E_FAIL,
1017 p->tr("Could not find data to snapshots '%s'"), d->snapshotId.toString().c_str());
1018
1019 if (d->mode == CloneMode_MachineState)
1020 {
1021 if (sn.uuid.isValid() && !sn.uuid.isZero())
1022 {
1023 trgMCF.hardwareMachine = sn.hardware;
1024 trgMCF.storageMachine = sn.storage;
1025 }
1026
1027 /* Remove any hint on snapshots. */
1028 trgMCF.llFirstSnapshot.clear();
1029 trgMCF.uuidCurrentSnapshot.clear();
1030 }
1031 else if ( d->mode == CloneMode_MachineAndChildStates
1032 && sn.uuid.isValid()
1033 && !sn.uuid.isZero())
1034 {
1035 if (!d->pOldMachineState.isNull())
1036 {
1037 /* Copy the snapshot data to the current machine. */
1038 trgMCF.hardwareMachine = sn.hardware;
1039 trgMCF.storageMachine = sn.storage;
1040
1041 /* Current state is under root snapshot. */
1042 trgMCF.uuidCurrentSnapshot = sn.uuid;
1043 }
1044 /* The snapshot will be the root one. */
1045 trgMCF.llFirstSnapshot.clear();
1046 trgMCF.llFirstSnapshot.push_back(sn);
1047 }
1048
1049 /* Generate new MAC addresses for all machines when not forbidden. */
1050 if (!d->options.contains(CloneOptions_KeepAllMACs))
1051 {
1052 d->updateMACAddresses(trgMCF.hardwareMachine.llNetworkAdapters);
1053 d->updateMACAddresses(trgMCF.llFirstSnapshot);
1054 }
1055
1056 /* When the current snapshot folder is absolute we reset it to the
1057 * default relative folder. */
1058 if (RTPathStartsWithRoot(trgMCF.machineUserData.strSnapshotFolder.c_str()))
1059 trgMCF.machineUserData.strSnapshotFolder = "Snapshots";
1060 trgMCF.strStateFile = "";
1061 /* Set the new name. */
1062 const Utf8Str strOldVMName = trgMCF.machineUserData.strName;
1063 trgMCF.machineUserData.strName = d->pTrgMachine->mUserData->s.strName;
1064 trgMCF.uuid = d->pTrgMachine->mData->mUuid;
1065
1066 Bstr bstrSrcSnapshotFolder;
1067 rc = d->pSrcMachine->COMGETTER(SnapshotFolder)(bstrSrcSnapshotFolder.asOutParam());
1068 if (FAILED(rc)) throw rc;
1069 /* The absolute name of the snapshot folder. */
1070 strTrgSnapshotFolder = Utf8StrFmt("%s%c%s", strTrgMachineFolder.c_str(), RTPATH_DELIMITER,
1071 trgMCF.machineUserData.strSnapshotFolder.c_str());
1072
1073 /* Should we rename the disk names. */
1074 bool fKeepDiskNames = d->options.contains(CloneOptions_KeepDiskNames);
1075
1076 /* We need to create a map with the already created medias. This is
1077 * necessary, cause different snapshots could have the same
1078 * parents/parent chain. If a medium is in this map already, it isn't
1079 * cloned a second time, but simply used. */
1080 typedef std::map<Utf8Str, ComObjPtr<Medium> > TStrMediumMap;
1081 typedef std::pair<Utf8Str, ComObjPtr<Medium> > TStrMediumPair;
1082 TStrMediumMap map;
1083 size_t cDisks = 0;
1084 for (size_t i = 0; i < d->llMedias.size(); ++i)
1085 {
1086 const MEDIUMTASKCHAIN &mtc = d->llMedias.at(i);
1087 ComObjPtr<Medium> pNewParent;
1088 uint32_t uSrcParentIdx = UINT32_MAX;
1089 uint32_t uTrgParentIdx = UINT32_MAX;
1090 for (size_t a = mtc.chain.size(); a > 0; --a)
1091 {
1092 const MEDIUMTASK &mt = mtc.chain.at(a - 1);
1093 ComPtr<IMedium> pMedium = mt.pMedium;
1094
1095 Bstr bstrSrcName;
1096 rc = pMedium->COMGETTER(Name)(bstrSrcName.asOutParam());
1097 if (FAILED(rc)) throw rc;
1098
1099 rc = d->pProgress->SetNextOperation(BstrFmt(p->tr("Cloning Disk '%ls' ..."), bstrSrcName.raw()).raw(),
1100 mt.uWeight);
1101 if (FAILED(rc)) throw rc;
1102
1103 Bstr bstrSrcId;
1104 rc = pMedium->COMGETTER(Id)(bstrSrcId.asOutParam());
1105 if (FAILED(rc)) throw rc;
1106
1107 if (mtc.fAttachLinked)
1108 {
1109 IMedium *pTmp = pMedium;
1110 ComObjPtr<Medium> pLMedium = static_cast<Medium*>(pTmp);
1111 if (pLMedium.isNull())
1112 throw p->setError(VBOX_E_OBJECT_NOT_FOUND);
1113 ComObjPtr<Medium> pBase = pLMedium->i_getBase();
1114 if (pBase->i_isReadOnly())
1115 {
1116 ComObjPtr<Medium> pDiff;
1117 /* create the diff under the snapshot medium */
1118 trgLock.release();
1119 srcLock.release();
1120 rc = d->createDifferencingMedium(p, pLMedium, strTrgSnapshotFolder,
1121 newMedia, &pDiff);
1122 srcLock.acquire();
1123 trgLock.acquire();
1124 if (FAILED(rc)) throw rc;
1125 map.insert(TStrMediumPair(Utf8Str(bstrSrcId), pDiff));
1126 /* diff image has to be used... */
1127 pNewParent = pDiff;
1128 }
1129 else
1130 {
1131 /* Attach the medium directly, as its type is not
1132 * subject to diff creation. */
1133 newMedia.append(pLMedium);
1134 map.insert(TStrMediumPair(Utf8Str(bstrSrcId), pLMedium));
1135 pNewParent = pLMedium;
1136 }
1137 }
1138 else
1139 {
1140 /* Is a clone already there? */
1141 TStrMediumMap::iterator it = map.find(Utf8Str(bstrSrcId));
1142 if (it != map.end())
1143 pNewParent = it->second;
1144 else
1145 {
1146 ComPtr<IMediumFormat> pSrcFormat;
1147 rc = pMedium->COMGETTER(MediumFormat)(pSrcFormat.asOutParam());
1148 ULONG uSrcCaps = 0;
1149 com::SafeArray <MediumFormatCapabilities_T> mediumFormatCap;
1150 rc = pSrcFormat->COMGETTER(Capabilities)(ComSafeArrayAsOutParam(mediumFormatCap));
1151
1152 if (FAILED(rc)) throw rc;
1153 else
1154 {
1155 for (ULONG j = 0; j < mediumFormatCap.size(); j++)
1156 uSrcCaps |= mediumFormatCap[j];
1157 }
1158
1159 /* Default format? */
1160 Utf8Str strDefaultFormat;
1161 if (mtc.devType == DeviceType_HardDisk)
1162 p->mParent->i_getDefaultHardDiskFormat(strDefaultFormat);
1163 else
1164 strDefaultFormat = "RAW";
1165
1166 Bstr bstrSrcFormat(strDefaultFormat);
1167
1168 ULONG srcVar = MediumVariant_Standard;
1169 com::SafeArray <MediumVariant_T> mediumVariant;
1170
1171 /* Is the source file based? */
1172 if ((uSrcCaps & MediumFormatCapabilities_File) == MediumFormatCapabilities_File)
1173 {
1174 /* Yes, just use the source format. Otherwise the defaults
1175 * will be used. */
1176 rc = pMedium->COMGETTER(Format)(bstrSrcFormat.asOutParam());
1177 if (FAILED(rc)) throw rc;
1178
1179 rc = pMedium->COMGETTER(Variant)(ComSafeArrayAsOutParam(mediumVariant));
1180 if (FAILED(rc)) throw rc;
1181 else
1182 {
1183 for (size_t j = 0; j < mediumVariant.size(); j++)
1184 srcVar |= mediumVariant[j];
1185 }
1186 }
1187
1188 Guid newId;
1189 newId.create();
1190 Utf8Str strNewName(bstrSrcName);
1191 if (!fKeepDiskNames)
1192 {
1193 Utf8Str strSrcTest = bstrSrcName;
1194 /* Check if we have to use another name. */
1195 if (!mt.strBaseName.isEmpty())
1196 strSrcTest = mt.strBaseName;
1197 strSrcTest.stripSuffix();
1198 /* If the old disk name was in {uuid} format we also
1199 * want the new name in this format, but with the
1200 * updated id of course. If the old disk was called
1201 * like the VM name, we change it to the new VM name.
1202 * For all other disks we rename them with this
1203 * template: "new name-disk1.vdi". */
1204 if (strSrcTest == strOldVMName)
1205 strNewName = Utf8StrFmt("%s%s", trgMCF.machineUserData.strName.c_str(),
1206 RTPathSuffix(Utf8Str(bstrSrcName).c_str()));
1207 else if ( strSrcTest.startsWith("{")
1208 && strSrcTest.endsWith("}"))
1209 {
1210 strSrcTest = strSrcTest.substr(1, strSrcTest.length() - 2);
1211
1212 Guid temp_guid(strSrcTest);
1213 if (temp_guid.isValid() && !temp_guid.isZero())
1214 strNewName = Utf8StrFmt("%s%s", newId.toStringCurly().c_str(),
1215 RTPathSuffix(strNewName.c_str()));
1216 }
1217 else
1218 strNewName = Utf8StrFmt("%s-disk%d%s", trgMCF.machineUserData.strName.c_str(), ++cDisks,
1219 RTPathSuffix(Utf8Str(bstrSrcName).c_str()));
1220 }
1221
1222 /* Check if this medium comes from the snapshot folder, if
1223 * so, put it there in the cloned machine as well.
1224 * Otherwise it goes to the machine folder. */
1225 Bstr bstrSrcPath;
1226 Utf8Str strFile = Utf8StrFmt("%s%c%s", strTrgMachineFolder.c_str(), RTPATH_DELIMITER, strNewName.c_str());
1227 rc = pMedium->COMGETTER(Location)(bstrSrcPath.asOutParam());
1228 if (FAILED(rc)) throw rc;
1229 if ( !bstrSrcPath.isEmpty()
1230 && RTPathStartsWith(Utf8Str(bstrSrcPath).c_str(), Utf8Str(bstrSrcSnapshotFolder).c_str())
1231 && (fKeepDiskNames || mt.strBaseName.isEmpty()))
1232 strFile = Utf8StrFmt("%s%c%s", strTrgSnapshotFolder.c_str(), RTPATH_DELIMITER, strNewName.c_str());
1233
1234 /* Start creating the clone. */
1235 ComObjPtr<Medium> pTarget;
1236 rc = pTarget.createObject();
1237 if (FAILED(rc)) throw rc;
1238
1239 rc = pTarget->init(p->mParent,
1240 Utf8Str(bstrSrcFormat),
1241 strFile,
1242 Guid::Empty /* empty media registry */,
1243 mtc.devType);
1244 if (FAILED(rc)) throw rc;
1245
1246 /* Update the new uuid. */
1247 pTarget->i_updateId(newId);
1248
1249 /* Do the disk cloning. */
1250 ComPtr<IProgress> progress2;
1251
1252 ComObjPtr<Medium> pLMedium = static_cast<Medium*>((IMedium*)pMedium);
1253 srcLock.release();
1254 rc = pLMedium->i_cloneToEx(pTarget,
1255 srcVar,
1256 pNewParent,
1257 progress2.asOutParam(),
1258 uSrcParentIdx,
1259 uTrgParentIdx);
1260 srcLock.acquire();
1261 if (FAILED(rc)) throw rc;
1262
1263 /* Wait until the async process has finished. */
1264 srcLock.release();
1265 rc = d->pProgress->WaitForAsyncProgressCompletion(progress2);
1266 srcLock.acquire();
1267 if (FAILED(rc)) throw rc;
1268
1269 /* Check the result of the async process. */
1270 LONG iRc;
1271 rc = progress2->COMGETTER(ResultCode)(&iRc);
1272 if (FAILED(rc)) throw rc;
1273 /* If the thread of the progress object has an error, then
1274 * retrieve the error info from there, or it'll be lost. */
1275 if (FAILED(iRc))
1276 throw p->setError(ProgressErrorInfo(progress2));
1277 /* Remember created medium. */
1278 newMedia.append(pTarget);
1279 /* Get the medium type from the source and set it to the
1280 * new medium. */
1281 MediumType_T type;
1282 rc = pMedium->COMGETTER(Type)(&type);
1283 if (FAILED(rc)) throw rc;
1284 trgLock.release();
1285 srcLock.release();
1286 rc = pTarget->COMSETTER(Type)(type);
1287 srcLock.acquire();
1288 trgLock.acquire();
1289 if (FAILED(rc)) throw rc;
1290 map.insert(TStrMediumPair(Utf8Str(bstrSrcId), pTarget));
1291 /* register the new medium */
1292 {
1293 AutoWriteLock tlock(p->mParent->i_getMediaTreeLockHandle() COMMA_LOCKVAL_SRC_POS);
1294 rc = p->mParent->i_registerMedium(pTarget, &pTarget,
1295 tlock);
1296 if (FAILED(rc)) throw rc;
1297 }
1298 /* This medium becomes the parent of the next medium in the
1299 * chain. */
1300 pNewParent = pTarget;
1301 }
1302 }
1303 /* Save the current source medium index as the new parent
1304 * medium index. */
1305 uSrcParentIdx = mt.uIdx;
1306 /* Simply increase the target index. */
1307 ++uTrgParentIdx;
1308 }
1309
1310 Bstr bstrSrcId;
1311 rc = mtc.chain.first().pMedium->COMGETTER(Id)(bstrSrcId.asOutParam());
1312 if (FAILED(rc)) throw rc;
1313 Bstr bstrTrgId;
1314 rc = pNewParent->COMGETTER(Id)(bstrTrgId.asOutParam());
1315 if (FAILED(rc)) throw rc;
1316 /* update snapshot configuration */
1317 d->updateSnapshotStorageLists(trgMCF.llFirstSnapshot, bstrSrcId, bstrTrgId);
1318
1319 /* create new 'Current State' diff for caller defined place */
1320 if (mtc.fCreateDiffs)
1321 {
1322 const MEDIUMTASK &mt = mtc.chain.first();
1323 ComObjPtr<Medium> pLMedium = static_cast<Medium*>((IMedium*)mt.pMedium);
1324 if (pLMedium.isNull())
1325 throw p->setError(VBOX_E_OBJECT_NOT_FOUND);
1326 ComObjPtr<Medium> pBase = pLMedium->i_getBase();
1327 if (pBase->i_isReadOnly())
1328 {
1329 ComObjPtr<Medium> pDiff;
1330 trgLock.release();
1331 srcLock.release();
1332 rc = d->createDifferencingMedium(p, pNewParent, strTrgSnapshotFolder,
1333 newMedia, &pDiff);
1334 srcLock.acquire();
1335 trgLock.acquire();
1336 if (FAILED(rc)) throw rc;
1337 /* diff image has to be used... */
1338 pNewParent = pDiff;
1339 }
1340 else
1341 {
1342 /* Attach the medium directly, as its type is not
1343 * subject to diff creation. */
1344 newMedia.append(pNewParent);
1345 }
1346
1347 rc = pNewParent->COMGETTER(Id)(bstrTrgId.asOutParam());
1348 if (FAILED(rc)) throw rc;
1349 }
1350 /* update 'Current State' configuration */
1351 d->updateStorageLists(trgMCF.storageMachine.llStorageControllers, bstrSrcId, bstrTrgId);
1352 }
1353 /* Make sure all disks know of the new machine uuid. We do this last to
1354 * be able to change the medium type above. */
1355 for (size_t i = newMedia.size(); i > 0; --i)
1356 {
1357 const ComObjPtr<Medium> &pMedium = newMedia.at(i - 1);
1358 AutoCaller mac(pMedium);
1359 if (FAILED(mac.rc())) throw mac.rc();
1360 AutoWriteLock mlock(pMedium COMMA_LOCKVAL_SRC_POS);
1361 Guid uuid = d->pTrgMachine->mData->mUuid;
1362 if (d->options.contains(CloneOptions_Link))
1363 {
1364 ComObjPtr<Medium> pParent = pMedium->i_getParent();
1365 mlock.release();
1366 if (!pParent.isNull())
1367 {
1368 AutoCaller mac2(pParent);
1369 if (FAILED(mac2.rc())) throw mac2.rc();
1370 AutoReadLock mlock2(pParent COMMA_LOCKVAL_SRC_POS);
1371 if (pParent->i_getFirstRegistryMachineId(uuid))
1372 {
1373 mlock2.release();
1374 trgLock.release();
1375 srcLock.release();
1376 p->mParent->i_markRegistryModified(uuid);
1377 srcLock.acquire();
1378 trgLock.acquire();
1379 mlock2.acquire();
1380 }
1381 }
1382 mlock.acquire();
1383 }
1384 pMedium->i_addRegistry(uuid);
1385 }
1386 /* Check if a snapshot folder is necessary and if so doesn't already
1387 * exists. */
1388 if ( !d->llSaveStateFiles.isEmpty()
1389 && !RTDirExists(strTrgSnapshotFolder.c_str()))
1390 {
1391 int vrc = RTDirCreateFullPath(strTrgSnapshotFolder.c_str(), 0700);
1392 if (RT_FAILURE(vrc))
1393 throw p->setError(VBOX_E_IPRT_ERROR,
1394 p->tr("Could not create snapshots folder '%s' (%Rrc)"),
1395 strTrgSnapshotFolder.c_str(), vrc);
1396 }
1397 /* Clone all save state files. */
1398 for (size_t i = 0; i < d->llSaveStateFiles.size(); ++i)
1399 {
1400 SAVESTATETASK sst = d->llSaveStateFiles.at(i);
1401 const Utf8Str &strTrgSaveState = Utf8StrFmt("%s%c%s", strTrgSnapshotFolder.c_str(), RTPATH_DELIMITER,
1402 RTPathFilename(sst.strSaveStateFile.c_str()));
1403
1404 /* Move to next sub-operation. */
1405 rc = d->pProgress->SetNextOperation(BstrFmt(p->tr("Copy save state file '%s' ..."),
1406 RTPathFilename(sst.strSaveStateFile.c_str())).raw(), sst.uWeight);
1407 if (FAILED(rc)) throw rc;
1408 /* Copy the file only if it was not copied already. */
1409 if (!newFiles.contains(strTrgSaveState.c_str()))
1410 {
1411 int vrc = RTFileCopyEx(sst.strSaveStateFile.c_str(), strTrgSaveState.c_str(), 0,
1412 MachineCloneVMPrivate::copyStateFileProgress, &d->pProgress);
1413 if (RT_FAILURE(vrc))
1414 throw p->setError(VBOX_E_IPRT_ERROR,
1415 p->tr("Could not copy state file '%s' to '%s' (%Rrc)"),
1416 sst.strSaveStateFile.c_str(), strTrgSaveState.c_str(), vrc);
1417 newFiles.append(strTrgSaveState);
1418 }
1419 /* Update the path in the configuration either for the current
1420 * machine state or the snapshots. */
1421 if (!sst.snapshotUuid.isValid() || sst.snapshotUuid.isZero())
1422 trgMCF.strStateFile = strTrgSaveState;
1423 else
1424 d->updateStateFile(trgMCF.llFirstSnapshot, sst.snapshotUuid, strTrgSaveState);
1425 }
1426
1427 {
1428 rc = d->pProgress->SetNextOperation(BstrFmt(p->tr("Create Machine Clone '%s' ..."),
1429 trgMCF.machineUserData.strName.c_str()).raw(), 1);
1430 if (FAILED(rc)) throw rc;
1431 /* After modifying the new machine config, we can copy the stuff
1432 * over to the new machine. The machine have to be mutable for
1433 * this. */
1434 rc = d->pTrgMachine->i_checkStateDependency(p->MutableStateDep);
1435 if (FAILED(rc)) throw rc;
1436 rc = d->pTrgMachine->i_loadMachineDataFromSettings(trgMCF, &d->pTrgMachine->mData->mUuid);
1437 if (FAILED(rc)) throw rc;
1438
1439 /* Fix up the "current state modified" flag to what it should be,
1440 * as the value guessed in i_loadMachineDataFromSettings can be
1441 * quite far off the logical value for the cloned VM. */
1442 if (d->mode == CloneMode_MachineState)
1443 d->pTrgMachine->mData->mCurrentStateModified = FALSE;
1444 else if ( d->mode == CloneMode_MachineAndChildStates
1445 && sn.uuid.isValid()
1446 && !sn.uuid.isZero())
1447 {
1448 if (!d->pOldMachineState.isNull())
1449 {
1450 /* There will be created a new differencing image based on
1451 * this snapshot. So reset the modified state. */
1452 d->pTrgMachine->mData->mCurrentStateModified = FALSE;
1453 }
1454 else
1455 d->pTrgMachine->mData->mCurrentStateModified = p->mData->mCurrentStateModified;
1456 }
1457 else if (d->mode == CloneMode_AllStates)
1458 d->pTrgMachine->mData->mCurrentStateModified = p->mData->mCurrentStateModified;
1459
1460 /* If the target machine has saved state we MUST adjust the machine
1461 * state, otherwise saving settings will drop the information. */
1462 if (trgMCF.strStateFile.isNotEmpty())
1463 d->pTrgMachine->i_setMachineState(MachineState_Saved);
1464
1465 /* save all VM data */
1466 bool fNeedsGlobalSaveSettings = false;
1467 rc = d->pTrgMachine->i_saveSettings(&fNeedsGlobalSaveSettings, Machine::SaveS_Force);
1468 if (FAILED(rc)) throw rc;
1469 /* Release all locks */
1470 trgLock.release();
1471 srcLock.release();
1472 if (fNeedsGlobalSaveSettings)
1473 {
1474 /* save the global settings; for that we should hold only the
1475 * VirtualBox lock */
1476 AutoWriteLock vlock(p->mParent COMMA_LOCKVAL_SRC_POS);
1477 rc = p->mParent->i_saveSettings();
1478 if (FAILED(rc)) throw rc;
1479 }
1480 }
1481
1482 /* Any additional machines need saving? */
1483 p->mParent->i_saveModifiedRegistries();
1484 }
1485 catch (HRESULT rc2)
1486 {
1487 /* Error handling code only works correctly without locks held. */
1488 trgLock.release();
1489 srcLock.release();
1490 rc = rc2;
1491 }
1492 catch (...)
1493 {
1494 rc = VirtualBoxBase::handleUnexpectedExceptions(p, RT_SRC_POS);
1495 }
1496
1497 MultiResult mrc(rc);
1498 /* Cleanup on failure (CANCEL also) */
1499 if (FAILED(rc))
1500 {
1501 int vrc = VINF_SUCCESS;
1502 /* Delete all created files. */
1503 for (size_t i = 0; i < newFiles.size(); ++i)
1504 {
1505 vrc = RTFileDelete(newFiles.at(i).c_str());
1506 if (RT_FAILURE(vrc))
1507 mrc = p->setError(VBOX_E_IPRT_ERROR, p->tr("Could not delete file '%s' (%Rrc)"), newFiles.at(i).c_str(), vrc);
1508 }
1509 /* Delete all already created medias. (Reverse, cause there could be
1510 * parent->child relations.) */
1511 for (size_t i = newMedia.size(); i > 0; --i)
1512 {
1513 const ComObjPtr<Medium> &pMedium = newMedia.at(i - 1);
1514 mrc = pMedium->i_deleteStorage(NULL /* aProgress */,
1515 true /* aWait */);
1516 pMedium->Close();
1517 }
1518 /* Delete the snapshot folder when not empty. */
1519 if (!strTrgSnapshotFolder.isEmpty())
1520 RTDirRemove(strTrgSnapshotFolder.c_str());
1521 /* Delete the machine folder when not empty. */
1522 RTDirRemove(strTrgMachineFolder.c_str());
1523
1524 /* Must save the modified registries */
1525 p->mParent->i_saveModifiedRegistries();
1526 }
1527
1528 return mrc;
1529}
1530
1531void MachineCloneVM::destroy()
1532{
1533 delete this;
1534}
1535
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