VirtualBox

source: vbox/trunk/src/VBox/Main/ConsoleImplTeleporter.cpp@ 24873

Last change on this file since 24873 was 24770, checked in by vboxsync, 15 years ago

Main: fixed teleportation cancellation.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 40.3 KB
Line 
1/* $Id: ConsoleImplTeleporter.cpp 24770 2009-11-18 18:16:21Z vboxsync $ */
2/** @file
3 * VBox Console COM Class implementation, The Teleporter Part.
4 */
5
6/*
7 * Copyright (C) 2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#include "ConsoleImpl.h"
27#include "Global.h"
28#include "Logging.h"
29#include "ProgressImpl.h"
30
31#include <iprt/err.h>
32#include <iprt/rand.h>
33#include <iprt/tcp.h>
34#include <iprt/timer.h>
35
36#include <VBox/vmapi.h>
37#include <VBox/ssm.h>
38#include <VBox/err.h>
39#include <VBox/version.h>
40#include <VBox/com/string.h>
41
42
43/*******************************************************************************
44* Structures and Typedefs *
45*******************************************************************************/
46/**
47 * Base class for the teleporter state.
48 *
49 * These classes are used as advanced structs, not as proper classes.
50 */
51class TeleporterState
52{
53public:
54 ComPtr<Console> mptrConsole;
55 PVM mpVM;
56 ComObjPtr<Progress> mptrProgress;
57 Utf8Str mstrPassword;
58 bool const mfIsSource;
59
60 /** @name stream stuff
61 * @{ */
62 RTSOCKET mhSocket;
63 uint64_t moffStream;
64 uint32_t mcbReadBlock;
65 bool volatile mfStopReading;
66 bool volatile mfEndOfStream;
67 bool volatile mfIOError;
68 /** @} */
69
70 TeleporterState(Console *pConsole, PVM pVM, Progress *pProgress, bool fIsSource)
71 : mptrConsole(pConsole)
72 , mpVM(pVM)
73 , mptrProgress(pProgress)
74 , mfIsSource(fIsSource)
75 , mhSocket(NIL_RTSOCKET)
76 , moffStream(UINT64_MAX / 2)
77 , mcbReadBlock(0)
78 , mfStopReading(false)
79 , mfEndOfStream(false)
80 , mfIOError(false)
81 {
82 }
83};
84
85
86/**
87 * Teleporter state used by the source side.
88 */
89class TeleporterStateSrc : public TeleporterState
90{
91public:
92 Utf8Str mstrHostname;
93 uint32_t muPort;
94 MachineState_T menmOldMachineState;
95 bool mfSuspendedByUs;
96 bool mfUnlockedMedia;
97
98 TeleporterStateSrc(Console *pConsole, PVM pVM, Progress *pProgress, MachineState_T enmOldMachineState)
99 : TeleporterState(pConsole, pVM, pProgress, true /*fIsSource*/)
100 , muPort(UINT32_MAX)
101 , menmOldMachineState(enmOldMachineState)
102 , mfSuspendedByUs(false)
103 , mfUnlockedMedia(false)
104 {
105 }
106};
107
108
109/**
110 * Teleporter state used by the destiation side.
111 */
112class TeleporterStateTrg : public TeleporterState
113{
114public:
115 IMachine *mpMachine;
116 IInternalMachineControl *mpControl;
117 PRTTCPSERVER mhServer;
118 PRTTIMERLR mphTimerLR;
119 bool mfLockedMedia;
120 int mRc;
121
122 TeleporterStateTrg(Console *pConsole, PVM pVM, Progress *pProgress,
123 IMachine *pMachine, IInternalMachineControl *pControl,
124 PRTTIMERLR phTimerLR, bool fStartPaused)
125 : TeleporterState(pConsole, pVM, pProgress, false /*fIsSource*/)
126 , mpMachine(pMachine)
127 , mpControl(pControl)
128 , mhServer(NULL)
129 , mphTimerLR(phTimerLR)
130 , mfLockedMedia(false)
131 , mRc(VINF_SUCCESS)
132 {
133 }
134};
135
136
137/**
138 * TCP stream header.
139 *
140 * This is an extra layer for fixing the problem with figuring out when the SSM
141 * stream ends.
142 */
143typedef struct TELEPORTERTCPHDR
144{
145 /** Magic value. */
146 uint32_t u32Magic;
147 /** The size of the data block following this header.
148 * 0 indicates the end of the stream. */
149 uint32_t cb;
150} TELEPORTERTCPHDR;
151/** Magic value for TELEPORTERTCPHDR::u32Magic. (Egberto Gismonti Amin) */
152#define TELEPORTERTCPHDR_MAGIC UINT32_C(0x19471205)
153/** The max block size. */
154#define TELEPORTERTCPHDR_MAX_SIZE UINT32_C(0x00fffff8)
155
156
157/*******************************************************************************
158* Global Variables *
159*******************************************************************************/
160static const char g_szWelcome[] = "VirtualBox-Teleporter-1.0\n";
161
162
163/**
164 * Reads a string from the socket.
165 *
166 * @returns VBox status code.
167 *
168 * @param pState The teleporter state structure.
169 * @param pszBuf The output buffer.
170 * @param cchBuf The size of the output buffer.
171 *
172 */
173static int teleporterTcpReadLine(TeleporterState *pState, char *pszBuf, size_t cchBuf)
174{
175 char *pszStart = pszBuf;
176 RTSOCKET Sock = pState->mhSocket;
177
178 AssertReturn(cchBuf > 1, VERR_INTERNAL_ERROR);
179 *pszBuf = '\0';
180
181 /* dead simple approach. */
182 for (;;)
183 {
184 char ch;
185 int rc = RTTcpRead(Sock, &ch, sizeof(ch), NULL);
186 if (RT_FAILURE(rc))
187 {
188 LogRel(("Teleporter: RTTcpRead -> %Rrc while reading string ('%s')\n", rc, pszStart));
189 return rc;
190 }
191 if ( ch == '\n'
192 || ch == '\0')
193 return VINF_SUCCESS;
194 if (cchBuf <= 1)
195 {
196 LogRel(("Teleporter: String buffer overflow: '%s'\n", pszStart));
197 return VERR_BUFFER_OVERFLOW;
198 }
199 *pszBuf++ = ch;
200 *pszBuf = '\0';
201 cchBuf--;
202 }
203}
204
205
206/**
207 * Reads an ACK or NACK.
208 *
209 * @returns S_OK on ACK, E_FAIL+setError() on failure or NACK.
210 * @param pState The teleporter source state.
211 * @param pszWhich Which ACK is this this?
212 * @param pszNAckMsg Optional NACK message.
213 *
214 * @remarks the setError laziness forces this to be a Console member.
215 */
216HRESULT
217Console::teleporterSrcReadACK(TeleporterStateSrc *pState, const char *pszWhich,
218 const char *pszNAckMsg /*= NULL*/)
219{
220 char szMsg[128];
221 int vrc = teleporterTcpReadLine(pState, szMsg, sizeof(szMsg));
222 if (RT_FAILURE(vrc))
223 return setError(E_FAIL, tr("Failed reading ACK(%s): %Rrc"), pszWhich, vrc);
224 if (strcmp(szMsg, "ACK"))
225 {
226 if (!strncmp(szMsg, "NACK=", sizeof("NACK=") - 1))
227 {
228 int32_t vrc2;
229 vrc = RTStrToInt32Full(&szMsg[sizeof("NACK=") - 1], 10, &vrc2);
230 if (vrc == VINF_SUCCESS)
231 {
232 if (pszNAckMsg)
233 {
234 LogRel(("Teleporter: %s: NACK=%Rrc (%d)\n", pszWhich, vrc2, vrc2));
235 return setError(E_FAIL, pszNAckMsg);
236 }
237 return setError(E_FAIL, "NACK(%s) - %Rrc (%d)", pszWhich, vrc2, vrc2);
238 }
239 }
240 return setError(E_FAIL, tr("%s: Expected ACK or NACK, got '%s'"), pszWhich, szMsg);
241 }
242 return S_OK;
243}
244
245
246/**
247 * Submitts a command to the destination and waits for the ACK.
248 *
249 * @returns S_OK on ACKed command, E_FAIL+setError() on failure.
250 *
251 * @param pState The teleporter source state.
252 * @param pszCommand The command.
253 * @param fWaitForAck Whether to wait for the ACK.
254 *
255 * @remarks the setError laziness forces this to be a Console member.
256 */
257HRESULT
258Console::teleporterSrcSubmitCommand(TeleporterStateSrc *pState, const char *pszCommand, bool fWaitForAck /*= true*/)
259{
260 size_t cchCommand = strlen(pszCommand);
261 int vrc = RTTcpWrite(pState->mhSocket, pszCommand, cchCommand);
262 if (RT_SUCCESS(vrc))
263 vrc = RTTcpWrite(pState->mhSocket, "\n", sizeof("\n") - 1);
264 if (RT_SUCCESS(vrc))
265 vrc = RTTcpFlush(pState->mhSocket);
266 if (RT_FAILURE(vrc))
267 return setError(E_FAIL, tr("Failed writing command '%s': %Rrc"), pszCommand, vrc);
268 if (!fWaitForAck)
269 return S_OK;
270 return teleporterSrcReadACK(pState, pszCommand);
271}
272
273
274/**
275 * @copydoc SSMSTRMOPS::pfnWrite
276 */
277static DECLCALLBACK(int) teleporterTcpOpWrite(void *pvUser, uint64_t offStream, const void *pvBuf, size_t cbToWrite)
278{
279 TeleporterState *pState = (TeleporterState *)pvUser;
280
281 AssertReturn(cbToWrite > 0, VINF_SUCCESS);
282 AssertReturn(cbToWrite < UINT32_MAX, VERR_OUT_OF_RANGE);
283 AssertReturn(pState->mfIsSource, VERR_INVALID_HANDLE);
284
285 /* Poll for incoming NACKs and errors from the other side */
286 int rc = RTTcpSelectOne(pState->mhSocket, 0);
287 if (rc != VERR_TIMEOUT)
288 {
289 if (RT_SUCCESS(rc))
290 {
291 LogRel(("Teleporter/TCP: Incoming data found before write, assuming it's a cancel.\n"));
292 rc = VERR_SSM_CANCELLED;
293 }
294 else
295 LogRel(("Teleporter/TCP: RTTcpSelectOne -> %Rrc before write.\n", rc));
296 return rc;
297 }
298
299 for (;;)
300 {
301 /* Write block header. */
302 TELEPORTERTCPHDR Hdr;
303 Hdr.u32Magic = TELEPORTERTCPHDR_MAGIC;
304 Hdr.cb = RT_MIN((uint32_t)cbToWrite, TELEPORTERTCPHDR_MAX_SIZE);
305 rc = RTTcpWrite(pState->mhSocket, &Hdr, sizeof(Hdr));
306 if (RT_FAILURE(rc))
307 {
308 LogRel(("Teleporter/TCP: Header write error: %Rrc\n", rc));
309 return rc;
310 }
311
312 /* Write the data. */
313 rc = RTTcpWrite(pState->mhSocket, pvBuf, Hdr.cb);
314 if (RT_FAILURE(rc))
315 {
316 LogRel(("Teleporter/TCP: Data write error: %Rrc (cb=%#x)\n", rc, Hdr.cb));
317 return rc;
318 }
319 pState->moffStream += Hdr.cb;
320 if (Hdr.cb == cbToWrite)
321 return VINF_SUCCESS;
322
323 /* advance */
324 cbToWrite -= Hdr.cb;
325 pvBuf = (uint8_t const *)pvBuf + Hdr.cb;
326 }
327}
328
329
330/**
331 * Selects and poll for close condition.
332 *
333 * We can use a relatively high poll timeout here since it's only used to get
334 * us out of error paths. In the normal cause of events, we'll get a
335 * end-of-stream header.
336 *
337 * @returns VBox status code.
338 *
339 * @param pState The teleporter state data.
340 */
341static int teleporterTcpReadSelect(TeleporterState *pState)
342{
343 int rc;
344 do
345 {
346 rc = RTTcpSelectOne(pState->mhSocket, 1000);
347 if (RT_FAILURE(rc) && rc != VERR_TIMEOUT)
348 {
349 pState->mfIOError = true;
350 LogRel(("Teleporter/TCP: Header select error: %Rrc\n", rc));
351 break;
352 }
353 if (pState->mfStopReading)
354 {
355 rc = VERR_EOF;
356 break;
357 }
358 } while (rc == VERR_TIMEOUT);
359 return rc;
360}
361
362
363/**
364 * @copydoc SSMSTRMOPS::pfnRead
365 */
366static DECLCALLBACK(int) teleporterTcpOpRead(void *pvUser, uint64_t offStream, void *pvBuf, size_t cbToRead, size_t *pcbRead)
367{
368 TeleporterState *pState = (TeleporterState *)pvUser;
369 AssertReturn(!pState->mfIsSource, VERR_INVALID_HANDLE);
370
371 for (;;)
372 {
373 int rc;
374
375 /*
376 * Check for various conditions and may have been signalled.
377 */
378 if (pState->mfEndOfStream)
379 return VERR_EOF;
380 if (pState->mfStopReading)
381 return VERR_EOF;
382 if (pState->mfIOError)
383 return VERR_IO_GEN_FAILURE;
384
385 /*
386 * If there is no more data in the current block, read the next
387 * block header.
388 */
389 if (!pState->mcbReadBlock)
390 {
391 rc = teleporterTcpReadSelect(pState);
392 if (RT_FAILURE(rc))
393 return rc;
394 TELEPORTERTCPHDR Hdr;
395 rc = RTTcpRead(pState->mhSocket, &Hdr, sizeof(Hdr), NULL);
396 if (RT_FAILURE(rc))
397 {
398 pState->mfIOError = true;
399 LogRel(("Teleporter/TCP: Header read error: %Rrc\n", rc));
400 return rc;
401 }
402 if ( Hdr.u32Magic != TELEPORTERTCPHDR_MAGIC
403 || Hdr.cb > TELEPORTERTCPHDR_MAX_SIZE)
404 {
405 pState->mfIOError = true;
406 LogRel(("Teleporter/TCP: Invalid block: u32Magic=%#x cb=%#x\n", Hdr.u32Magic, Hdr.cb));
407 return VERR_IO_GEN_FAILURE;
408 }
409
410 pState->mcbReadBlock = Hdr.cb;
411 if (!Hdr.cb)
412 {
413 pState->mfEndOfStream = true;
414 return VERR_EOF;
415 }
416
417 if (pState->mfStopReading)
418 return VERR_EOF;
419 }
420
421 /*
422 * Read more data.
423 */
424 rc = teleporterTcpReadSelect(pState);
425 if (RT_FAILURE(rc))
426 return rc;
427 uint32_t cb = (uint32_t)RT_MIN(pState->mcbReadBlock, cbToRead);
428 rc = RTTcpRead(pState->mhSocket, pvBuf, cb, pcbRead);
429 if (RT_FAILURE(rc))
430 {
431 pState->mfIOError = true;
432 LogRel(("Teleporter/TCP: Data read error: %Rrc (cb=%#x)\n", rc, cb));
433 return rc;
434 }
435 if (pcbRead)
436 {
437 cb = (uint32_t)*pcbRead;
438 pState->moffStream += cb;
439 pState->mcbReadBlock -= cb;
440 return VINF_SUCCESS;
441 }
442 pState->moffStream += cb;
443 pState->mcbReadBlock -= cb;
444 if (cbToRead == cb)
445 return VINF_SUCCESS;
446
447 /* Advance to the next block. */
448 cbToRead -= cb;
449 pvBuf = (uint8_t *)pvBuf + cb;
450 }
451}
452
453
454/**
455 * @copydoc SSMSTRMOPS::pfnSeek
456 */
457static DECLCALLBACK(int) teleporterTcpOpSeek(void *pvUser, int64_t offSeek, unsigned uMethod, uint64_t *poffActual)
458{
459 return VERR_NOT_SUPPORTED;
460}
461
462
463/**
464 * @copydoc SSMSTRMOPS::pfnTell
465 */
466static DECLCALLBACK(uint64_t) teleporterTcpOpTell(void *pvUser)
467{
468 TeleporterState *pState = (TeleporterState *)pvUser;
469 return pState->moffStream;
470}
471
472
473/**
474 * @copydoc SSMSTRMOPS::pfnSize
475 */
476static DECLCALLBACK(int) teleporterTcpOpSize(void *pvUser, uint64_t *pcb)
477{
478 return VERR_NOT_SUPPORTED;
479}
480
481
482/**
483 * @copydoc SSMSTRMOPS::pfnClose
484 */
485static DECLCALLBACK(int) teleporterTcpOpClose(void *pvUser)
486{
487 TeleporterState *pState = (TeleporterState *)pvUser;
488
489 if (pState->mfIsSource)
490 {
491 TELEPORTERTCPHDR EofHdr = { TELEPORTERTCPHDR_MAGIC, 0 };
492 int rc = RTTcpWrite(pState->mhSocket, &EofHdr, sizeof(EofHdr));
493 if (RT_SUCCESS(rc))
494 rc = RTTcpFlush(pState->mhSocket);
495 if (RT_FAILURE(rc))
496 {
497 LogRel(("Teleporter/TCP: EOF Header write error: %Rrc\n", rc));
498 return rc;
499 }
500 }
501 else
502 {
503 ASMAtomicWriteBool(&pState->mfStopReading, true);
504 RTTcpFlush(pState->mhSocket);
505 }
506
507 return VINF_SUCCESS;
508}
509
510
511/**
512 * Method table for a TCP based stream.
513 */
514static SSMSTRMOPS const g_teleporterTcpOps =
515{
516 SSMSTRMOPS_VERSION,
517 teleporterTcpOpWrite,
518 teleporterTcpOpRead,
519 teleporterTcpOpSeek,
520 teleporterTcpOpTell,
521 teleporterTcpOpSize,
522 teleporterTcpOpClose,
523 SSMSTRMOPS_VERSION
524};
525
526
527/**
528 * Progress cancelation callback.
529 */
530static void teleporterProgressCancelCallback(void *pvUser)
531{
532 TeleporterState *pState = (TeleporterState *)pvUser;
533 SSMR3Cancel(pState->mpVM);
534 if (!pState->mfIsSource)
535 {
536 TeleporterStateTrg *pStateTrg = (TeleporterStateTrg *)pState;
537 RTTcpServerShutdown(pStateTrg->mhServer);
538 }
539}
540
541/**
542 * @copydoc PFNVMPROGRESS
543 */
544static DECLCALLBACK(int) teleporterProgressCallback(PVM pVM, unsigned uPercent, void *pvUser)
545{
546 TeleporterState *pState = (TeleporterState *)pvUser;
547 if (pState->mptrProgress)
548 {
549 HRESULT hrc = pState->mptrProgress->SetCurrentOperationProgress(uPercent);
550 if (FAILED(hrc))
551 {
552 /* check if the failure was caused by cancellation. */
553 BOOL fCancelled;
554 hrc = pState->mptrProgress->COMGETTER(Canceled)(&fCancelled);
555 if (SUCCEEDED(hrc) && fCancelled)
556 {
557 SSMR3Cancel(pState->mpVM);
558 return VERR_SSM_CANCELLED;
559 }
560 }
561 }
562
563 return VINF_SUCCESS;
564}
565
566
567/**
568 * @copydoc FNRTTIMERLR
569 */
570static DECLCALLBACK(void) teleporterDstTimeout(RTTIMERLR hTimerLR, void *pvUser, uint64_t iTick)
571{
572 /* This is harmless for any open connections. */
573 RTTcpServerShutdown((PRTTCPSERVER)pvUser);
574}
575
576
577/**
578 * Do the teleporter.
579 *
580 * @returns VBox status code.
581 * @param pState The teleporter state.
582 */
583HRESULT
584Console::teleporterSrc(TeleporterStateSrc *pState)
585{
586 AutoCaller autoCaller(this);
587 CheckComRCReturnRC(autoCaller.rc());
588
589 /*
590 * Wait for Console::Teleport to change the state.
591 */
592 { AutoWriteLock autoLock(this); }
593
594 BOOL fCancelled = TRUE;
595 HRESULT hrc = pState->mptrProgress->COMGETTER(Canceled)(&fCancelled);
596 if (FAILED(hrc))
597 return hrc;
598 if (fCancelled)
599 return setError(E_FAIL, tr("cancelled"));
600
601 /*
602 * Try connect to the destination machine.
603 * (Note. The caller cleans up mhSocket, so we can return without worries.)
604 */
605 int vrc = RTTcpClientConnect(pState->mstrHostname.c_str(), pState->muPort, &pState->mhSocket);
606 if (RT_FAILURE(vrc))
607 return setError(E_FAIL, tr("Failed to connect to port %u on '%s': %Rrc"),
608 pState->muPort, pState->mstrHostname.c_str(), vrc);
609
610 /* Read and check the welcome message. */
611 char szLine[RT_MAX(128, sizeof(g_szWelcome))];
612 RT_ZERO(szLine);
613 vrc = RTTcpRead(pState->mhSocket, szLine, sizeof(g_szWelcome) - 1, NULL);
614 if (RT_FAILURE(vrc))
615 return setError(E_FAIL, tr("Failed to read welcome message: %Rrc"), vrc);
616 if (strcmp(szLine, g_szWelcome))
617 return setError(E_FAIL, tr("Unexpected welcome %.*Rhxs"), sizeof(g_szWelcome) - 1, szLine);
618
619 /* password */
620 pState->mstrPassword.append('\n');
621 vrc = RTTcpWrite(pState->mhSocket, pState->mstrPassword.c_str(), pState->mstrPassword.length());
622 if (RT_FAILURE(vrc))
623 return setError(E_FAIL, tr("Failed to send password: %Rrc"), vrc);
624
625 /* ACK */
626 hrc = teleporterSrcReadACK(pState, "password", tr("Invalid password"));
627 if (FAILED(hrc))
628 return hrc;
629
630 /*
631 * Start loading the state.
632 *
633 * Note! The saved state includes vital configuration data which will be
634 * verified against the VM config on the other end. This is all done
635 * in the first pass, so we should fail pretty promptly on misconfig.
636 */
637 hrc = teleporterSrcSubmitCommand(pState, "load");
638 if (FAILED(hrc))
639 return hrc;
640
641 void *pvUser = static_cast<void *>(static_cast<TeleporterState *>(pState));
642 vrc = VMR3Teleport(pState->mpVM, &g_teleporterTcpOps, pvUser, teleporterProgressCallback, pvUser, &pState->mfSuspendedByUs);
643 if (RT_FAILURE(vrc))
644 return setError(E_FAIL, tr("VMR3Teleport -> %Rrc"), vrc);
645
646 hrc = teleporterSrcReadACK(pState, "load-complete");
647 if (FAILED(hrc))
648 return hrc;
649
650 /*
651 * We're at the point of no return.
652 */
653 if (!pState->mptrProgress->notifyPointOfNoReturn())
654 {
655 teleporterSrcSubmitCommand(pState, "cancel", false /*fWaitForAck*/);
656 return E_FAIL;
657 }
658
659 /*
660 * Hand over any media which we might be sharing.
661 *
662 * Note! This is only important on localhost teleportations.
663 */
664 /** @todo Maybe we should only do this if it's a local teleportation... */
665 hrc = mControl->UnlockMedia();
666 if (FAILED(hrc))
667 return hrc;
668 pState->mfUnlockedMedia = true;
669
670 hrc = teleporterSrcSubmitCommand(pState, "lock-media");
671 if (FAILED(hrc))
672 return hrc;
673
674 /*
675 * The FINAL step is giving the target instructions how to proceed with the VM.
676 */
677 if ( vrc == VINF_SSM_LIVE_SUSPENDED
678 || pState->menmOldMachineState == MachineState_Paused)
679 hrc = teleporterSrcSubmitCommand(pState, "hand-over-paused");
680 else
681 hrc = teleporterSrcSubmitCommand(pState, "hand-over-resume");
682 if (FAILED(hrc))
683 return hrc;
684
685 /*
686 * teleporterSrcThreadWrapper will do the automatic power off because it
687 * has to release the AutoVMCaller.
688 */
689 return S_OK;
690}
691
692
693/**
694 * Static thread method wrapper.
695 *
696 * @returns VINF_SUCCESS (ignored).
697 * @param hThread The thread.
698 * @param pvUser Pointer to a TeleporterStateSrc instance.
699 */
700/*static*/ DECLCALLBACK(int)
701Console::teleporterSrcThreadWrapper(RTTHREAD hThread, void *pvUser)
702{
703 TeleporterStateSrc *pState = (TeleporterStateSrc *)pvUser;
704
705 /*
706 * Console::teleporterSrc does the work, we just grab onto the VM handle
707 * and do the cleanups afterwards.
708 */
709 AutoVMCaller autoVMCaller(pState->mptrConsole);
710 HRESULT hrc = autoVMCaller.rc();
711
712 if (SUCCEEDED(hrc))
713 hrc = pState->mptrConsole->teleporterSrc(pState);
714
715 /* Close the connection ASAP on so that the other side can complete. */
716 if (pState->mhSocket != NIL_RTSOCKET)
717 {
718 RTTcpClientClose(pState->mhSocket);
719 pState->mhSocket = NIL_RTSOCKET;
720 }
721
722 /* Aaarg! setMachineState trashes error info on Windows, so we have to
723 complete things here on failure instead of right before cleanup. */
724 if (FAILED(hrc))
725 pState->mptrProgress->notifyComplete(hrc);
726
727 /* We can no longer be cancelled (success), or it doesn't matter any longer (failure). */
728 pState->mptrProgress->setCancelCallback(NULL, NULL);
729
730 /*
731 * Write lock the console before resetting mptrCancelableProgress and
732 * fixing the state.
733 */
734 AutoWriteLock autoLock(pState->mptrConsole);
735 pState->mptrConsole->mptrCancelableProgress.setNull();
736
737 VMSTATE const enmVMState = VMR3GetState(pState->mpVM);
738 MachineState_T const enmMachineState = pState->mptrConsole->mMachineState;
739 if (SUCCEEDED(hrc))
740 {
741 /*
742 * Automatically shut down the VM on success.
743 *
744 * Note! We have to release the VM caller object or we'll deadlock in
745 * powerDown.
746 */
747 AssertLogRelMsg(enmVMState == VMSTATE_SUSPENDED, ("%s\n", VMR3GetStateName(enmVMState)));
748 AssertLogRelMsg(enmMachineState == MachineState_TeleportingPausedVM, ("%s\n", Global::stringifyMachineState(enmMachineState)));
749
750 autoVMCaller.release();
751 hrc = pState->mptrConsole->powerDown();
752 pState->mptrProgress->notifyComplete(hrc);
753 }
754 else
755 {
756 /*
757 * Work the state machinery on failure.
758 *
759 * If the state is no longer 'Teleporting*', some other operation has
760 * canceled us and there is nothing we need to do here. In all other
761 * cases, we've failed one way or another.
762 */
763 if ( enmMachineState == MachineState_Teleporting
764 || enmMachineState == MachineState_TeleportingPausedVM
765 )
766 {
767 if (pState->mfUnlockedMedia)
768 {
769 ErrorInfoKeeper Oak;
770 HRESULT hrc2 = pState->mptrConsole->mControl->LockMedia();
771 if (FAILED(hrc2))
772 {
773 uint64_t StartMS = RTTimeMilliTS();
774 do
775 {
776 RTThreadSleep(2);
777 hrc2 = pState->mptrConsole->mControl->LockMedia();
778 } while ( FAILED(hrc2)
779 && RTTimeMilliTS() - StartMS < 2000);
780 }
781 if (SUCCEEDED(hrc2))
782 pState->mfUnlockedMedia = true;
783 else
784 LogRel(("FATAL ERROR: Failed to re-take the media locks. hrc2=%Rhrc\n", hrc2));
785 }
786
787 switch (enmVMState)
788 {
789 case VMSTATE_RUNNING:
790 case VMSTATE_RUNNING_LS:
791 case VMSTATE_DEBUGGING:
792 case VMSTATE_DEBUGGING_LS:
793 case VMSTATE_POWERING_OFF:
794 case VMSTATE_POWERING_OFF_LS:
795 case VMSTATE_RESETTING:
796 case VMSTATE_RESETTING_LS:
797 Assert(!pState->mfSuspendedByUs);
798 Assert(!pState->mfUnlockedMedia);
799 pState->mptrConsole->setMachineState(MachineState_Running);
800 break;
801
802 case VMSTATE_GURU_MEDITATION:
803 case VMSTATE_GURU_MEDITATION_LS:
804 pState->mptrConsole->setMachineState(MachineState_Stuck);
805 break;
806
807 case VMSTATE_FATAL_ERROR:
808 case VMSTATE_FATAL_ERROR_LS:
809 pState->mptrConsole->setMachineState(MachineState_Paused);
810 break;
811
812 default:
813 AssertMsgFailed(("%s\n", VMR3GetStateName(enmVMState)));
814 case VMSTATE_SUSPENDED:
815 case VMSTATE_SUSPENDED_LS:
816 case VMSTATE_SUSPENDING:
817 case VMSTATE_SUSPENDING_LS:
818 case VMSTATE_SUSPENDING_EXT_LS:
819 if (!pState->mfUnlockedMedia)
820 {
821 pState->mptrConsole->setMachineState(MachineState_Paused);
822 if (pState->mfSuspendedByUs)
823 {
824 autoLock.leave();
825 int rc = VMR3Resume(pState->mpVM);
826 AssertLogRelMsgRC(rc, ("VMR3Resume -> %Rrc\n", rc));
827 autoLock.enter();
828 }
829 }
830 else
831 {
832 /* Faking a guru meditation is the best I can think of doing here... */
833 pState->mptrConsole->setMachineState(MachineState_Stuck);
834 }
835 break;
836 }
837 }
838 }
839 autoLock.leave();
840
841 /*
842 * Cleanup.
843 */
844 Assert(pState->mhSocket == NIL_RTSOCKET);
845 delete pState;
846
847 return VINF_SUCCESS; /* ignored */
848}
849
850
851/**
852 * Start teleporter to the specified target.
853 *
854 * @returns COM status code.
855 *
856 * @param aHostname The name of the target host.
857 * @param aPort The TCP port number.
858 * @param aPassword The password.
859 * @param aProgress Where to return the progress object.
860 */
861STDMETHODIMP
862Console::Teleport(IN_BSTR aHostname, ULONG aPort, IN_BSTR aPassword, IProgress **aProgress)
863{
864 /*
865 * Validate parameters, check+hold object status, write lock the object
866 * and validate the state.
867 */
868 CheckComArgOutPointerValid(aProgress);
869 CheckComArgStrNotEmptyOrNull(aHostname);
870 CheckComArgNotNull(aHostname);
871 CheckComArgExprMsg(aPort, aPort > 0 && aPort <= 65535, ("is %u", aPort));
872
873 AutoCaller autoCaller(this);
874 CheckComRCReturnRC(autoCaller.rc());
875
876 AutoWriteLock autoLock(this);
877 LogFlowThisFunc(("mMachineState=%d\n", mMachineState));
878
879 switch (mMachineState)
880 {
881 case MachineState_Running:
882 case MachineState_Paused:
883 break;
884
885 default:
886 return setError(VBOX_E_INVALID_VM_STATE,
887 tr("Invalid machine state: %s (must be Running or Paused)"),
888 Global::stringifyMachineState(mMachineState));
889 }
890
891
892 /*
893 * Create a progress object, spawn a worker thread and change the state.
894 * Note! The thread won't start working until we release the lock.
895 */
896 LogFlowThisFunc(("Initiating TELEPORT request...\n"));
897
898 ComObjPtr<Progress> ptrProgress;
899 HRESULT hrc = ptrProgress.createObject();
900 CheckComRCReturnRC(hrc);
901 hrc = ptrProgress->init(static_cast<IConsole *>(this), Bstr(tr("Teleporter")), TRUE /*aCancelable*/);
902 CheckComRCReturnRC(hrc);
903
904 TeleporterStateSrc *pState = new TeleporterStateSrc(this, mpVM, ptrProgress, mMachineState);
905 pState->mstrPassword = aPassword;
906 pState->mstrHostname = aHostname;
907 pState->muPort = aPort;
908
909 void *pvUser = static_cast<void *>(static_cast<TeleporterState *>(pState));
910 ptrProgress->setCancelCallback(teleporterProgressCancelCallback, pvUser);
911
912 int vrc = RTThreadCreate(NULL, Console::teleporterSrcThreadWrapper, (void *)pState, 0 /*cbStack*/,
913 RTTHREADTYPE_EMULATION, 0 /*fFlags*/, "Teleport");
914 if (RT_SUCCESS(vrc))
915 {
916 if (mMachineState == MachineState_Running)
917 hrc = setMachineState(MachineState_Teleporting);
918 else
919 hrc = setMachineState(MachineState_TeleportingPausedVM);
920 if (SUCCEEDED(hrc))
921 {
922 ptrProgress.queryInterfaceTo(aProgress);
923 mptrCancelableProgress = ptrProgress;
924 }
925 else
926 ptrProgress->Cancel();
927 }
928 else
929 {
930 ptrProgress->setCancelCallback(NULL, NULL);
931 delete pState;
932 hrc = setError(E_FAIL, tr("RTThreadCreate -> %Rrc"), vrc);
933 }
934
935 return hrc;
936}
937
938
939/**
940 * Creates a TCP server that listens for the source machine and passes control
941 * over to Console::teleporterTrgServeConnection().
942 *
943 * @returns VBox status code.
944 * @param pVM The VM handle
945 * @param pMachine The IMachine for the virtual machine.
946 * @param fStartPaused Whether to start it in the Paused (true) or
947 * Running (false) state,
948 * @param pProgress Pointer to the progress object.
949 *
950 * @remarks The caller expects error information to be set on failure.
951 * @todo Check that all the possible failure paths sets error info...
952 */
953int
954Console::teleporterTrg(PVM pVM, IMachine *pMachine, bool fStartPaused, Progress *pProgress)
955{
956 /*
957 * Get the config.
958 */
959 ULONG uPort;
960 HRESULT hrc = pMachine->COMGETTER(TeleporterPort)(&uPort);
961 if (FAILED(hrc))
962 return VERR_GENERAL_FAILURE;
963 ULONG const uPortOrg = uPort;
964
965 Bstr bstrAddress;
966 hrc = pMachine->COMGETTER(TeleporterAddress)(bstrAddress.asOutParam());
967 if (FAILED(hrc))
968 return VERR_GENERAL_FAILURE;
969 Utf8Str strAddress(bstrAddress);
970 const char *pszAddress = strAddress.isEmpty() ? NULL : strAddress.c_str();
971
972 Bstr bstrPassword;
973 hrc = pMachine->COMGETTER(TeleporterPassword)(bstrPassword.asOutParam());
974 if (FAILED(hrc))
975 return VERR_GENERAL_FAILURE;
976 Utf8Str strPassword(bstrPassword);
977 strPassword.append('\n'); /* To simplify password checking. */
978
979 /*
980 * Create the TCP server.
981 */
982 int vrc;
983 PRTTCPSERVER hServer;
984 if (uPort)
985 vrc = RTTcpServerCreateEx(pszAddress, uPort, &hServer);
986 else
987 {
988 for (int cTries = 10240; cTries > 0; cTries--)
989 {
990 uPort = RTRandU32Ex(cTries >= 8192 ? 49152 : 1024, 65534);
991 vrc = RTTcpServerCreateEx(pszAddress, uPort, &hServer);
992 if (vrc != VERR_NET_ADDRESS_IN_USE)
993 break;
994 }
995 if (RT_SUCCESS(vrc))
996 {
997 hrc = pMachine->COMSETTER(TeleporterPort)(uPort);
998 if (FAILED(hrc))
999 {
1000 RTTcpServerDestroy(hServer);
1001 return VERR_GENERAL_FAILURE;
1002 }
1003 }
1004 }
1005 if (RT_FAILURE(vrc))
1006 return vrc;
1007
1008 /*
1009 * Create a one-shot timer for timing out after 5 mins.
1010 */
1011 RTTIMERLR hTimerLR;
1012 vrc = RTTimerLRCreateEx(&hTimerLR, 0 /*ns*/, RTTIMER_FLAGS_CPU_ANY, teleporterDstTimeout, hServer);
1013 if (RT_SUCCESS(vrc))
1014 {
1015 vrc = RTTimerLRStart(hTimerLR, 5*60*UINT64_C(1000000000) /*ns*/);
1016 if (RT_SUCCESS(vrc))
1017 {
1018 /*
1019 * Do the job, when it returns we're done.
1020 */
1021 TeleporterStateTrg State(this, pVM, pProgress, pMachine, mControl, &hTimerLR, fStartPaused);
1022 State.mstrPassword = strPassword;
1023 State.mhServer = hServer;
1024
1025 void *pvUser = static_cast<void *>(static_cast<TeleporterState *>(&State));
1026 if (pProgress->setCancelCallback(teleporterProgressCancelCallback, pvUser))
1027 {
1028 vrc = RTTcpServerListen(hServer, Console::teleporterTrgServeConnection, &State);
1029 pProgress->setCancelCallback(NULL, NULL);
1030
1031 bool fPowerOff = false;
1032 if (vrc == VERR_TCP_SERVER_STOP)
1033 {
1034 vrc = State.mRc;
1035 /* Power off the VM on failure unless the state callback
1036 already did that. */
1037 if (RT_FAILURE(vrc))
1038 {
1039 VMSTATE enmVMState = VMR3GetState(pVM);
1040 if ( enmVMState != VMSTATE_OFF
1041 && enmVMState != VMSTATE_POWERING_OFF)
1042 fPowerOff = true;
1043 }
1044 }
1045 else if (vrc == VERR_TCP_SERVER_SHUTDOWN)
1046 {
1047 BOOL fCancelled = TRUE;
1048 hrc = pProgress->COMGETTER(Canceled)(&fCancelled);
1049 if (FAILED(hrc) || fCancelled)
1050 {
1051 setError(E_FAIL, tr("Teleporting canceled"));
1052 vrc = VERR_SSM_CANCELLED;
1053 }
1054 else
1055 {
1056 setError(E_FAIL, tr("Teleporter timed out waiting for incoming connection"));
1057 vrc = VERR_TIMEOUT;
1058 }
1059 LogRel(("Teleporter: RTTcpServerListen aborted - %Rrc\n", vrc));
1060 fPowerOff = true;
1061 }
1062 else
1063 {
1064 LogRel(("Teleporter: Unexpected RTTcpServerListen rc: %Rrc\n", vrc));
1065 vrc = VERR_IPE_UNEXPECTED_STATUS;
1066 fPowerOff = true;
1067 }
1068
1069 if (fPowerOff)
1070 {
1071 int vrc2 = VMR3PowerOff(pVM);
1072 AssertRC(vrc2);
1073 }
1074 }
1075 else
1076 vrc = VERR_SSM_CANCELLED;
1077 }
1078
1079 RTTimerLRDestroy(hTimerLR);
1080 }
1081 RTTcpServerDestroy(hServer);
1082
1083 /*
1084 * If we change TeleporterPort above, set it back to it's original
1085 * value before returning.
1086 */
1087 if (uPortOrg != uPort)
1088 pMachine->COMSETTER(TeleporterPort)(uPortOrg);
1089
1090 return vrc;
1091}
1092
1093
1094/**
1095 * Unlock the media.
1096 *
1097 * This is used in error paths.
1098 *
1099 * @param pState The teleporter state.
1100 */
1101static void teleporterTrgUnlockMedia(TeleporterStateTrg *pState)
1102{
1103 if (pState->mfLockedMedia)
1104 {
1105 pState->mpControl->UnlockMedia();
1106 pState->mfLockedMedia = false;
1107 }
1108}
1109
1110
1111static int teleporterTcpWriteACK(TeleporterStateTrg *pState, bool fAutomaticUnlock = true)
1112{
1113 int rc = RTTcpWrite(pState->mhSocket, "ACK\n", sizeof("ACK\n") - 1);
1114 if (RT_FAILURE(rc))
1115 {
1116 LogRel(("Teleporter: RTTcpWrite(,ACK,) -> %Rrc\n", rc));
1117 if (fAutomaticUnlock)
1118 teleporterTrgUnlockMedia(pState);
1119 }
1120 RTTcpFlush(pState->mhSocket);
1121 return rc;
1122}
1123
1124
1125static int teleporterTcpWriteNACK(TeleporterStateTrg *pState, int32_t rc2)
1126{
1127 /*
1128 * Unlock media sending the NACK. That way the other doesn't have to spin
1129 * waiting to regain the locks.
1130 */
1131 teleporterTrgUnlockMedia(pState);
1132
1133 char szMsg[64];
1134 size_t cch = RTStrPrintf(szMsg, sizeof(szMsg), "NACK=%d\n", rc2);
1135 int rc = RTTcpWrite(pState->mhSocket, szMsg, cch);
1136 if (RT_FAILURE(rc))
1137 LogRel(("Teleporter: RTTcpWrite(,%s,%zu) -> %Rrc\n", szMsg, cch, rc));
1138 RTTcpFlush(pState->mhSocket);
1139 return rc;
1140}
1141
1142
1143/**
1144 * @copydoc FNRTTCPSERVE
1145 *
1146 * @returns VINF_SUCCESS or VERR_TCP_SERVER_STOP.
1147 */
1148/*static*/ DECLCALLBACK(int)
1149Console::teleporterTrgServeConnection(RTSOCKET Sock, void *pvUser)
1150{
1151 TeleporterStateTrg *pState = (TeleporterStateTrg *)pvUser;
1152 pState->mhSocket = Sock;
1153
1154 /*
1155 * Say hello.
1156 */
1157 int vrc = RTTcpWrite(Sock, g_szWelcome, sizeof(g_szWelcome) - 1);
1158 if (RT_FAILURE(vrc))
1159 {
1160 LogRel(("Teleporter: Failed to write welcome message: %Rrc\n", vrc));
1161 return VINF_SUCCESS;
1162 }
1163
1164 /*
1165 * Password (includes '\n', see teleporterTrg).
1166 */
1167 const char *pszPassword = pState->mstrPassword.c_str();
1168 unsigned off = 0;
1169 while (pszPassword[off])
1170 {
1171 char ch;
1172 vrc = RTTcpRead(Sock, &ch, sizeof(ch), NULL);
1173 if ( RT_FAILURE(vrc)
1174 || pszPassword[off] != ch)
1175 {
1176 if (RT_FAILURE(vrc))
1177 LogRel(("Teleporter: Password read failure (off=%u): %Rrc\n", off, vrc));
1178 else
1179 LogRel(("Teleporter: Invalid password (off=%u)\n", off));
1180 teleporterTcpWriteNACK(pState, VERR_AUTHENTICATION_FAILURE);
1181 return VINF_SUCCESS;
1182 }
1183 off++;
1184 }
1185 vrc = teleporterTcpWriteACK(pState);
1186 if (RT_FAILURE(vrc))
1187 return VINF_SUCCESS;
1188
1189 /*
1190 * Stop the server and cancel the timeout timer.
1191 *
1192 * Note! After this point we must return VERR_TCP_SERVER_STOP, while prior
1193 * to it we must not return that value!
1194 */
1195 RTTcpServerShutdown(pState->mhServer);
1196 RTTimerLRDestroy(*pState->mphTimerLR);
1197 *pState->mphTimerLR = NIL_RTTIMERLR;
1198
1199 /*
1200 * Command processing loop.
1201 */
1202 bool fDone = false;
1203 for (;;)
1204 {
1205 char szCmd[128];
1206 vrc = teleporterTcpReadLine(pState, szCmd, sizeof(szCmd));
1207 if (RT_FAILURE(vrc))
1208 break;
1209
1210 if (!strcmp(szCmd, "load"))
1211 {
1212 vrc = teleporterTcpWriteACK(pState);
1213 if (RT_FAILURE(vrc))
1214 break;
1215
1216 pState->moffStream = 0;
1217 void *pvUser = static_cast<void *>(static_cast<TeleporterState *>(pState));
1218 vrc = VMR3LoadFromStream(pState->mpVM, &g_teleporterTcpOps, pvUser,
1219 teleporterProgressCallback, pvUser);
1220 if (RT_FAILURE(vrc))
1221 {
1222 LogRel(("Teleporter: VMR3LoadFromStream -> %Rrc\n", vrc));
1223 teleporterTcpWriteNACK(pState, vrc);
1224 break;
1225 }
1226
1227 /* The EOS might not have been read, make sure it is. */
1228 pState->mfStopReading = false;
1229 size_t cbRead;
1230 vrc = teleporterTcpOpRead(pvUser, pState->moffStream, szCmd, 1, &cbRead);
1231 if (vrc != VERR_EOF)
1232 {
1233 LogRel(("Teleporter: Draining teleporterTcpOpRead -> %Rrc\n", vrc));
1234 teleporterTcpWriteNACK(pState, vrc);
1235 break;
1236 }
1237
1238 vrc = teleporterTcpWriteACK(pState);
1239 }
1240 else if (!strcmp(szCmd, "cancel"))
1241 {
1242 /* Don't ACK this. */
1243 LogRel(("Teleporter: Received cancel command.\n"));
1244 vrc = VERR_SSM_CANCELLED;
1245 }
1246 else if (!strcmp(szCmd, "lock-media"))
1247 {
1248 HRESULT hrc = pState->mpControl->LockMedia();
1249 if (SUCCEEDED(hrc))
1250 {
1251 pState->mfLockedMedia = true;
1252 vrc = teleporterTcpWriteACK(pState);
1253 }
1254 else
1255 {
1256 vrc = VERR_FILE_LOCK_FAILED;
1257 teleporterTcpWriteNACK(pState, vrc);
1258 }
1259 }
1260 else if ( !strcmp(szCmd, "hand-over-resume")
1261 || !strcmp(szCmd, "hand-over-paused"))
1262 {
1263 /*
1264 * Point of no return.
1265 *
1266 * Note! Since we cannot tell whether a VMR3Resume failure is
1267 * destructive for the source or not, we have little choice
1268 * but to ACK it first and take any failures locally.
1269 *
1270 * Ideally, we should try resume it first and then ACK (or
1271 * NACK) the request since this would reduce latency and
1272 * make it possible to recover from some VMR3Resume failures.
1273 */
1274 if ( pState->mptrProgress->notifyPointOfNoReturn()
1275 && pState->mfLockedMedia)
1276 {
1277 vrc = teleporterTcpWriteACK(pState);
1278 if (RT_SUCCESS(vrc))
1279 {
1280 if (!strcmp(szCmd, "hand-over-resume"))
1281 vrc = VMR3Resume(pState->mpVM);
1282 else
1283 pState->mptrConsole->setMachineState(MachineState_Paused);
1284 fDone = true;
1285 break;
1286 }
1287 }
1288 else
1289 {
1290 vrc = pState->mfLockedMedia ? VERR_WRONG_ORDER : VERR_SSM_CANCELLED;
1291 teleporterTcpWriteNACK(pState, vrc);
1292 }
1293 }
1294 else
1295 {
1296 LogRel(("Teleporter: Unknown command '%s' (%.*Rhxs)\n", szCmd, strlen(szCmd), szCmd));
1297 vrc = VERR_NOT_IMPLEMENTED;
1298 teleporterTcpWriteNACK(pState, vrc);
1299 }
1300
1301 if (RT_FAILURE(vrc))
1302 break;
1303 }
1304
1305 if (RT_SUCCESS(vrc) && !fDone)
1306 vrc = VERR_WRONG_ORDER;
1307 if (RT_FAILURE(vrc))
1308 teleporterTrgUnlockMedia(pState);
1309
1310 pState->mRc = vrc;
1311 pState->mhSocket = NIL_RTSOCKET;
1312 LogFlowFunc(("returns mRc=%Rrc\n", vrc));
1313 return VERR_TCP_SERVER_STOP;
1314}
1315
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