VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/VRDEServerImpl.cpp@ 61362

Last change on this file since 61362 was 61009, checked in by vboxsync, 9 years ago

Main: big settings cleanup and writing optimization. Moved constructors/equality/default checks into the .cpp file, and write only settings which aren't at the default value. Greatly reduces the effort needed to write everything out, especially when a lot of snapshots have to be dealt with. Move the storage controllers to the hardware settings, where they always belonged. No change to the XML file (yet). Lots of settings related cleanups in the API code.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 25.4 KB
Line 
1/* $Id: VRDEServerImpl.cpp 61009 2016-05-17 17:18:29Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox COM class implementation
5 */
6
7/*
8 * Copyright (C) 2006-2016 Oracle Corporation
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19#include "VRDEServerImpl.h"
20#include "MachineImpl.h"
21#include "VirtualBoxImpl.h"
22#ifdef VBOX_WITH_EXTPACK
23# include "ExtPackManagerImpl.h"
24#endif
25
26#include <iprt/cpp/utils.h>
27#include <iprt/ctype.h>
28#include <iprt/ldr.h>
29#include <iprt/path.h>
30
31#include <VBox/err.h>
32#include <VBox/sup.h>
33#include <VBox/com/array.h>
34
35#include <VBox/RemoteDesktop/VRDE.h>
36
37#include "AutoStateDep.h"
38#include "AutoCaller.h"
39#include "Global.h"
40#include "Logging.h"
41
42// defines
43/////////////////////////////////////////////////////////////////////////////
44#define VRDP_DEFAULT_PORT_STR "3389"
45
46// constructor / destructor
47/////////////////////////////////////////////////////////////////////////////
48
49VRDEServer::VRDEServer()
50 : mParent(NULL)
51{
52}
53
54VRDEServer::~VRDEServer()
55{
56}
57
58HRESULT VRDEServer::FinalConstruct()
59{
60 return BaseFinalConstruct();
61}
62
63void VRDEServer::FinalRelease()
64{
65 uninit();
66 BaseFinalRelease();
67}
68
69// public initializer/uninitializer for internal purposes only
70/////////////////////////////////////////////////////////////////////////////
71
72/**
73 * Initializes the VRDP server object.
74 *
75 * @param aParent Handle of the parent object.
76 */
77HRESULT VRDEServer::init(Machine *aParent)
78{
79 LogFlowThisFunc(("aParent=%p\n", aParent));
80
81 ComAssertRet(aParent, E_INVALIDARG);
82
83 /* Enclose the state transition NotReady->InInit->Ready */
84 AutoInitSpan autoInitSpan(this);
85 AssertReturn(autoInitSpan.isOk(), E_FAIL);
86
87 unconst(mParent) = aParent;
88 /* mPeer is left null */
89
90 mData.allocate();
91
92 /* Confirm a successful initialization */
93 autoInitSpan.setSucceeded();
94
95 return S_OK;
96}
97
98/**
99 * Initializes the object given another object
100 * (a kind of copy constructor). This object shares data with
101 * the object passed as an argument.
102 *
103 * @note This object must be destroyed before the original object
104 * it shares data with is destroyed.
105 *
106 * @note Locks @a aThat object for reading.
107 */
108HRESULT VRDEServer::init(Machine *aParent, VRDEServer *aThat)
109{
110 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
111
112 ComAssertRet(aParent && aThat, E_INVALIDARG);
113
114 /* Enclose the state transition NotReady->InInit->Ready */
115 AutoInitSpan autoInitSpan(this);
116 AssertReturn(autoInitSpan.isOk(), E_FAIL);
117
118 unconst(mParent) = aParent;
119 unconst(mPeer) = aThat;
120
121 AutoCaller thatCaller(aThat);
122 AssertComRCReturnRC(thatCaller.rc());
123
124 AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
125 mData.share(aThat->mData);
126
127 /* Confirm a successful initialization */
128 autoInitSpan.setSucceeded();
129
130 return S_OK;
131}
132
133/**
134 * Initializes the guest object given another guest object
135 * (a kind of copy constructor). This object makes a private copy of data
136 * of the original object passed as an argument.
137 *
138 * @note Locks @a aThat object for reading.
139 */
140HRESULT VRDEServer::initCopy(Machine *aParent, VRDEServer *aThat)
141{
142 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
143
144 ComAssertRet(aParent && aThat, E_INVALIDARG);
145
146 /* Enclose the state transition NotReady->InInit->Ready */
147 AutoInitSpan autoInitSpan(this);
148 AssertReturn(autoInitSpan.isOk(), E_FAIL);
149
150 unconst(mParent) = aParent;
151 /* mPeer is left null */
152
153 AutoCaller thatCaller(aThat);
154 AssertComRCReturnRC(thatCaller.rc());
155
156 AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
157 mData.attachCopy(aThat->mData);
158
159 /* Confirm a successful initialization */
160 autoInitSpan.setSucceeded();
161
162 return S_OK;
163}
164
165/**
166 * Uninitializes the instance and sets the ready flag to FALSE.
167 * Called either from FinalRelease() or by the parent when it gets destroyed.
168 */
169void VRDEServer::uninit()
170{
171 LogFlowThisFunc(("\n"));
172
173 /* Enclose the state transition Ready->InUninit->NotReady */
174 AutoUninitSpan autoUninitSpan(this);
175 if (autoUninitSpan.uninitDone())
176 return;
177
178 mData.free();
179
180 unconst(mPeer) = NULL;
181 unconst(mParent) = NULL;
182}
183
184/**
185 * Loads settings from the given machine node.
186 * May be called once right after this object creation.
187 *
188 * @param aMachineNode <Machine> node.
189 *
190 * @note Locks this object for writing.
191 */
192HRESULT VRDEServer::i_loadSettings(const settings::VRDESettings &data)
193{
194 using namespace settings;
195
196 AutoCaller autoCaller(this);
197 AssertComRCReturnRC(autoCaller.rc());
198
199 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
200 mData.assignCopy(&data);
201
202 return S_OK;
203}
204
205/**
206 * Saves settings to the given machine node.
207 *
208 * @param aMachineNode <Machine> node.
209 *
210 * @note Locks this object for reading.
211 */
212HRESULT VRDEServer::i_saveSettings(settings::VRDESettings &data)
213{
214 AutoCaller autoCaller(this);
215 AssertComRCReturnRC(autoCaller.rc());
216
217 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
218 data = *mData.data();
219
220 return S_OK;
221}
222
223// IVRDEServer properties
224/////////////////////////////////////////////////////////////////////////////
225
226HRESULT VRDEServer::getEnabled(BOOL *aEnabled)
227{
228 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
229
230 *aEnabled = mData->fEnabled;
231
232 return S_OK;
233}
234
235HRESULT VRDEServer::setEnabled(BOOL aEnabled)
236{
237 /* the machine can also be in saved state for this property to change */
238 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
239 if (FAILED(adep.rc())) return adep.rc();
240
241 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
242
243 HRESULT rc = S_OK;
244
245 if (mData->fEnabled != RT_BOOL(aEnabled))
246 {
247 mData.backup();
248 mData->fEnabled = RT_BOOL(aEnabled);
249
250 /* leave the lock before informing callbacks */
251 alock.release();
252
253 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, needs no locking
254 mParent->i_setModified(Machine::IsModified_VRDEServer);
255 mlock.release();
256
257 /* Avoid deadlock when i_onVRDEServerChange eventually calls SetExtraData. */
258 adep.release();
259
260 rc = mParent->i_onVRDEServerChange(/* aRestart */ TRUE);
261 }
262
263 return rc;
264}
265
266static int i_portParseNumber(uint16_t *pu16Port, const char *pszStart, const char *pszEnd)
267{
268 /* Gets a string of digits, converts to 16 bit port number.
269 * Note: pszStart <= pszEnd is expected, the string contains
270 * only digits and pszEnd points to the char after last
271 * digit.
272 */
273 size_t cch = pszEnd - pszStart;
274 if (cch > 0 && cch <= 5) /* Port is up to 5 decimal digits. */
275 {
276 unsigned uPort = 0;
277 while (pszStart != pszEnd)
278 {
279 uPort = uPort * 10 + *pszStart - '0';
280 pszStart++;
281 }
282
283 if (uPort != 0 && uPort < 0x10000)
284 {
285 if (pu16Port)
286 *pu16Port = (uint16_t)uPort;
287 return VINF_SUCCESS;
288 }
289 }
290
291 return VERR_INVALID_PARAMETER;
292}
293
294static int i_vrdpServerVerifyPortsString(com::Utf8Str portRange)
295{
296 const char *pszPortRange = portRange.c_str();
297
298 if (!pszPortRange || *pszPortRange == 0) /* Reject empty string. */
299 return VERR_INVALID_PARAMETER;
300
301 /* The string should be like "1000-1010,1020,2000-2003" */
302 while (*pszPortRange)
303 {
304 const char *pszStart = pszPortRange;
305 const char *pszDash = NULL;
306 const char *pszEnd = pszStart;
307
308 while (*pszEnd && *pszEnd != ',')
309 {
310 if (*pszEnd == '-')
311 {
312 if (pszDash != NULL)
313 return VERR_INVALID_PARAMETER; /* More than one '-'. */
314
315 pszDash = pszEnd;
316 }
317 else if (!RT_C_IS_DIGIT(*pszEnd))
318 return VERR_INVALID_PARAMETER;
319
320 pszEnd++;
321 }
322
323 /* Update the next range pointer. */
324 pszPortRange = pszEnd;
325 if (*pszPortRange == ',')
326 {
327 pszPortRange++;
328 }
329
330 /* A probably valid range. Verify and parse it. */
331 int rc;
332 if (pszDash)
333 {
334 rc = i_portParseNumber(NULL, pszStart, pszDash);
335 if (RT_SUCCESS(rc))
336 rc = i_portParseNumber(NULL, pszDash + 1, pszEnd);
337 }
338 else
339 rc = i_portParseNumber(NULL, pszStart, pszEnd);
340
341 if (RT_FAILURE(rc))
342 return rc;
343 }
344
345 return VINF_SUCCESS;
346}
347
348HRESULT VRDEServer::setVRDEProperty(const com::Utf8Str &aKey, const com::Utf8Str &aValue)
349{
350 LogFlowThisFunc(("\n"));
351
352 /* the machine can also be in saved state for this property to change */
353 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
354 if (FAILED(adep.rc())) return adep.rc();
355
356 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
357
358 /* Special processing for some "standard" properties. */
359 if (aKey == "TCP/Ports")
360 {
361 /* Verify the string. "0" means the default port. */
362 Utf8Str strPorts = aValue == "0"?
363 VRDP_DEFAULT_PORT_STR:
364 aValue;
365 int vrc = i_vrdpServerVerifyPortsString(strPorts);
366 if (RT_FAILURE(vrc))
367 return E_INVALIDARG;
368
369 if (strPorts != mData->mapProperties["TCP/Ports"])
370 {
371 /* Port value is not verified here because it is up to VRDP transport to
372 * use it. Specifying a wrong port number will cause a running server to
373 * stop. There is no fool proof here.
374 */
375 mData.backup();
376 mData->mapProperties["TCP/Ports"] = strPorts;
377
378 /* leave the lock before informing callbacks */
379 alock.release();
380
381 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, needs no locking
382 mParent->i_setModified(Machine::IsModified_VRDEServer);
383 mlock.release();
384
385 /* Avoid deadlock when i_onVRDEServerChange eventually calls SetExtraData. */
386 adep.release();
387
388 mParent->i_onVRDEServerChange(/* aRestart */ TRUE);
389 }
390 }
391 else
392 {
393 /* Generic properties processing.
394 * Look up the old value first; if nothing's changed then do nothing.
395 */
396 Utf8Str strOldValue;
397
398 settings::StringsMap::const_iterator it = mData->mapProperties.find(aKey);
399 if (it != mData->mapProperties.end())
400 strOldValue = it->second;
401
402 if (strOldValue != aValue)
403 {
404 if (aValue.isEmpty())
405 mData->mapProperties.erase(aKey);
406 else
407 mData->mapProperties[aKey] = aValue;
408
409 /* leave the lock before informing callbacks */
410 alock.release();
411
412 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS);
413 mParent->i_setModified(Machine::IsModified_VRDEServer);
414 mlock.release();
415
416 /* Avoid deadlock when i_onVRDEServerChange eventually calls SetExtraData. */
417 adep.release();
418
419 mParent->i_onVRDEServerChange(/* aRestart */ TRUE);
420 }
421 }
422
423 return S_OK;
424}
425
426HRESULT VRDEServer::getVRDEProperty(const com::Utf8Str &aKey, com::Utf8Str &aValue)
427{
428 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
429 settings::StringsMap::const_iterator it = mData->mapProperties.find(aKey);
430 if (it != mData->mapProperties.end())
431 aValue = it->second; // source is a Utf8Str
432 else if (aKey == "TCP/Ports")
433 aValue = VRDP_DEFAULT_PORT_STR;
434
435 return S_OK;
436}
437
438static int loadVRDELibrary(const char *pszLibraryName, RTLDRMOD *phmod, PFNVRDESUPPORTEDPROPERTIES *ppfn)
439{
440 int rc = VINF_SUCCESS;
441
442 RTLDRMOD hmod = NIL_RTLDRMOD;
443
444 RTERRINFOSTATIC ErrInfo;
445 RTErrInfoInitStatic(&ErrInfo);
446 if (RTPathHavePath(pszLibraryName))
447 rc = SUPR3HardenedLdrLoadPlugIn(pszLibraryName, &hmod, &ErrInfo.Core);
448 else
449 rc = SUPR3HardenedLdrLoadAppPriv(pszLibraryName, &hmod, RTLDRLOAD_FLAGS_LOCAL, &ErrInfo.Core);
450 if (RT_SUCCESS(rc))
451 {
452 rc = RTLdrGetSymbol(hmod, "VRDESupportedProperties", (void **)ppfn);
453
454 if (RT_FAILURE(rc) && rc != VERR_SYMBOL_NOT_FOUND)
455 LogRel(("VRDE: Error resolving symbol '%s', rc %Rrc.\n", "VRDESupportedProperties", rc));
456 }
457 else
458 {
459 if (RTErrInfoIsSet(&ErrInfo.Core))
460 LogRel(("VRDE: Error loading the library '%s': %s (%Rrc)\n", pszLibraryName, ErrInfo.Core.pszMsg, rc));
461 else
462 LogRel(("VRDE: Error loading the library '%s' rc = %Rrc.\n", pszLibraryName, rc));
463
464 hmod = NIL_RTLDRMOD;
465 }
466
467 if (RT_SUCCESS(rc))
468 {
469 *phmod = hmod;
470 }
471 else
472 {
473 if (hmod != NIL_RTLDRMOD)
474 {
475 RTLdrClose(hmod);
476 hmod = NIL_RTLDRMOD;
477 }
478 }
479
480 return rc;
481}
482
483HRESULT VRDEServer::getVRDEProperties(std::vector<com::Utf8Str> &aProperties)
484{
485 size_t cProperties = 0;
486 aProperties.resize(0);
487 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
488 if (!mData->fEnabled)
489 {
490 return S_OK;
491 }
492 alock.release();
493
494 /*
495 * Check that a VRDE extension pack name is set and resolve it into a
496 * library path.
497 */
498 Bstr bstrExtPack;
499 HRESULT hrc = COMGETTER(VRDEExtPack)(bstrExtPack.asOutParam());
500 Log(("VRDEPROP: get extpack hrc 0x%08X, isEmpty %d\n", hrc, bstrExtPack.isEmpty()));
501 if (FAILED(hrc))
502 return hrc;
503 if (bstrExtPack.isEmpty())
504 return E_FAIL;
505
506 Utf8Str strExtPack(bstrExtPack);
507 Utf8Str strVrdeLibrary;
508 int vrc = VINF_SUCCESS;
509 if (strExtPack.equals(VBOXVRDP_KLUDGE_EXTPACK_NAME))
510 strVrdeLibrary = "VBoxVRDP";
511 else
512 {
513#ifdef VBOX_WITH_EXTPACK
514 VirtualBox *pVirtualBox = mParent->i_getVirtualBox();
515 ExtPackManager *pExtPackMgr = pVirtualBox->i_getExtPackManager();
516 vrc = pExtPackMgr->i_getVrdeLibraryPathForExtPack(&strExtPack, &strVrdeLibrary);
517#else
518 vrc = VERR_FILE_NOT_FOUND;
519#endif
520 }
521 Log(("VRDEPROP: library get rc %Rrc\n", vrc));
522
523 if (RT_SUCCESS(vrc))
524 {
525 /*
526 * Load the VRDE library and start the server, if it is enabled.
527 */
528 PFNVRDESUPPORTEDPROPERTIES pfn = NULL;
529 RTLDRMOD hmod = NIL_RTLDRMOD;
530 vrc = loadVRDELibrary(strVrdeLibrary.c_str(), &hmod, &pfn);
531 Log(("VRDEPROP: load library [%s] rc %Rrc\n", strVrdeLibrary.c_str(), vrc));
532 if (RT_SUCCESS(vrc))
533 {
534 const char * const *papszNames = pfn();
535
536 if (papszNames)
537 {
538 size_t i;
539 for (i = 0; papszNames[i] != NULL; ++i)
540 {
541 cProperties++;
542 }
543 }
544 Log(("VRDEPROP: %d properties\n", cProperties));
545
546 if (cProperties > 0)
547 {
548 aProperties.resize(cProperties);
549 for (size_t i = 0; i < cProperties && papszNames[i] != NULL; ++i)
550 {
551 aProperties[i] = papszNames[i];
552 }
553 }
554
555 /* Do not forget to unload the library. */
556 RTLdrClose(hmod);
557 hmod = NIL_RTLDRMOD;
558 }
559 }
560
561 if (RT_FAILURE(vrc))
562 {
563 return E_FAIL;
564 }
565
566 return S_OK;
567}
568
569
570HRESULT VRDEServer::getAuthType(AuthType_T *aType)
571{
572 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
573
574 *aType = mData->authType;
575
576 return S_OK;
577}
578
579HRESULT VRDEServer::setAuthType(AuthType_T aType)
580{
581 /* the machine can also be in saved state for this property to change */
582 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
583 if (FAILED(adep.rc())) return adep.rc();
584
585 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
586
587 if (mData->authType != aType)
588 {
589 mData.backup();
590 mData->authType = aType;
591
592 /* leave the lock before informing callbacks */
593 alock.release();
594
595 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, needs no locking
596 mParent->i_setModified(Machine::IsModified_VRDEServer);
597 mlock.release();
598
599 mParent->i_onVRDEServerChange(/* aRestart */ TRUE);
600 }
601
602 return S_OK;
603}
604
605HRESULT VRDEServer::getAuthTimeout(ULONG *aTimeout)
606{
607 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
608
609 *aTimeout = mData->ulAuthTimeout;
610
611 return S_OK;
612}
613
614
615HRESULT VRDEServer::setAuthTimeout(ULONG aTimeout)
616{
617 /* the machine can also be in saved state for this property to change */
618 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
619 if (FAILED(adep.rc())) return adep.rc();
620
621 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
622
623 if (aTimeout != mData->ulAuthTimeout)
624 {
625 mData.backup();
626 mData->ulAuthTimeout = aTimeout;
627
628 /* leave the lock before informing callbacks */
629 alock.release();
630
631 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, needs no locking
632 mParent->i_setModified(Machine::IsModified_VRDEServer);
633 mlock.release();
634
635 /* sunlover 20060131: This setter does not require the notification
636 * really */
637#if 0
638 mParent->onVRDEServerChange();
639#endif
640 }
641
642 return S_OK;
643}
644
645HRESULT VRDEServer::getAuthLibrary(com::Utf8Str &aLibrary)
646{
647 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
648 aLibrary = mData->strAuthLibrary;
649 alock.release();
650
651 if (aLibrary.isEmpty())
652 {
653 /* Get the global setting. */
654 ComPtr<ISystemProperties> systemProperties;
655 HRESULT hrc = mParent->i_getVirtualBox()->COMGETTER(SystemProperties)(systemProperties.asOutParam());
656 if (SUCCEEDED(hrc))
657 {
658 Bstr strlib;
659 hrc = systemProperties->COMGETTER(VRDEAuthLibrary)(strlib.asOutParam());
660 if (SUCCEEDED(hrc))
661 aLibrary = Utf8Str(strlib).c_str();
662 }
663
664 if (FAILED(hrc))
665 return setError(hrc, "failed to query the library setting\n");
666 }
667
668 return S_OK;
669}
670
671
672HRESULT VRDEServer::setAuthLibrary(const com::Utf8Str &aLibrary)
673{
674 /* the machine can also be in saved state for this property to change */
675 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
676 if (FAILED(adep.rc())) return adep.rc();
677
678 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
679
680 if (mData->strAuthLibrary != aLibrary)
681 {
682 mData.backup();
683 mData->strAuthLibrary = aLibrary;
684
685 /* leave the lock before informing callbacks */
686 alock.release();
687
688 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS);
689 mParent->i_setModified(Machine::IsModified_VRDEServer);
690 mlock.release();
691
692 mParent->i_onVRDEServerChange(/* aRestart */ TRUE);
693 }
694
695 return S_OK;
696}
697
698
699HRESULT VRDEServer::getAllowMultiConnection(BOOL *aAllowMultiConnection)
700{
701 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
702
703 *aAllowMultiConnection = mData->fAllowMultiConnection;
704
705 return S_OK;
706}
707
708
709HRESULT VRDEServer::setAllowMultiConnection(BOOL aAllowMultiConnection)
710{
711 /* the machine can also be in saved state for this property to change */
712 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
713 if (FAILED(adep.rc())) return adep.rc();
714
715 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
716
717 if (mData->fAllowMultiConnection != RT_BOOL(aAllowMultiConnection))
718 {
719 mData.backup();
720 mData->fAllowMultiConnection = RT_BOOL(aAllowMultiConnection);
721
722 /* leave the lock before informing callbacks */
723 alock.release();
724
725 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, needs no locking
726 mParent->i_setModified(Machine::IsModified_VRDEServer);
727 mlock.release();
728
729 mParent->i_onVRDEServerChange(/* aRestart */ TRUE); // @todo does it need a restart?
730 }
731
732 return S_OK;
733}
734
735HRESULT VRDEServer::getReuseSingleConnection(BOOL *aReuseSingleConnection)
736{
737 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
738
739 *aReuseSingleConnection = mData->fReuseSingleConnection;
740
741 return S_OK;
742}
743
744
745HRESULT VRDEServer::setReuseSingleConnection(BOOL aReuseSingleConnection)
746{
747 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
748 if (FAILED(adep.rc())) return adep.rc();
749
750 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
751
752 if (mData->fReuseSingleConnection != RT_BOOL(aReuseSingleConnection))
753 {
754 mData.backup();
755 mData->fReuseSingleConnection = RT_BOOL(aReuseSingleConnection);
756
757 /* leave the lock before informing callbacks */
758 alock.release();
759
760 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, needs no locking
761 mParent->i_setModified(Machine::IsModified_VRDEServer);
762 mlock.release();
763
764 mParent->i_onVRDEServerChange(/* aRestart */ TRUE); // @todo needs a restart?
765 }
766
767 return S_OK;
768}
769
770HRESULT VRDEServer::getVRDEExtPack(com::Utf8Str &aExtPack)
771{
772 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
773 Utf8Str strExtPack = mData->strVrdeExtPack;
774 alock.release();
775 HRESULT hrc = S_OK;
776
777 if (strExtPack.isNotEmpty())
778 {
779 if (strExtPack.equals(VBOXVRDP_KLUDGE_EXTPACK_NAME))
780 hrc = S_OK;
781 else
782 {
783#ifdef VBOX_WITH_EXTPACK
784 ExtPackManager *pExtPackMgr = mParent->i_getVirtualBox()->i_getExtPackManager();
785 hrc = pExtPackMgr->i_checkVrdeExtPack(&strExtPack);
786#else
787 hrc = setError(E_FAIL, tr("Extension pack '%s' does not exist"), strExtPack.c_str());
788#endif
789 }
790 if (SUCCEEDED(hrc))
791 aExtPack = strExtPack;
792 }
793 else
794 {
795 /* Get the global setting. */
796 ComPtr<ISystemProperties> systemProperties;
797 hrc = mParent->i_getVirtualBox()->COMGETTER(SystemProperties)(systemProperties.asOutParam());
798 if (SUCCEEDED(hrc))
799 {
800 BSTR bstr;
801 hrc = systemProperties->COMGETTER(DefaultVRDEExtPack)(&bstr);
802 if (SUCCEEDED(hrc))
803 aExtPack = Utf8Str(bstr);
804 }
805 }
806 return hrc;
807}
808
809// public methods only for internal purposes
810/////////////////////////////////////////////////////////////////////////////
811HRESULT VRDEServer::setVRDEExtPack(const com::Utf8Str &aExtPack)
812{
813 HRESULT hrc = S_OK;
814 /* the machine can also be in saved state for this property to change */
815 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
816 hrc = adep.rc();
817 if (SUCCEEDED(hrc))
818 {
819 /*
820 * If not empty, check the specific extension pack.
821 */
822 if (!aExtPack.isEmpty())
823 {
824 if (aExtPack.equals(VBOXVRDP_KLUDGE_EXTPACK_NAME))
825 hrc = S_OK;
826 else
827 {
828#ifdef VBOX_WITH_EXTPACK
829 ExtPackManager *pExtPackMgr = mParent->i_getVirtualBox()->i_getExtPackManager();
830 hrc = pExtPackMgr->i_checkVrdeExtPack(&aExtPack);
831#else
832 hrc = setError(E_FAIL, tr("Extension pack '%s' does not exist"), aExtPack.c_str());
833#endif
834 }
835 }
836 if (SUCCEEDED(hrc))
837 {
838 /*
839 * Update the setting if there is an actual change, post an
840 * change event to trigger a VRDE server restart.
841 */
842 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
843 if (aExtPack != mData->strVrdeExtPack)
844 {
845 mData.backup();
846 mData->strVrdeExtPack = aExtPack;
847
848 /* leave the lock before informing callbacks */
849 alock.release();
850
851 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS);
852 mParent->i_setModified(Machine::IsModified_VRDEServer);
853 mlock.release();
854
855 mParent->i_onVRDEServerChange(/* aRestart */ TRUE);
856 }
857 }
858 }
859
860 return hrc;
861}
862
863// public methods only for internal purposes
864/////////////////////////////////////////////////////////////////////////////
865
866/**
867 * @note Locks this object for writing.
868 */
869void VRDEServer::i_rollback()
870{
871 /* sanity */
872 AutoCaller autoCaller(this);
873 AssertComRCReturnVoid(autoCaller.rc());
874
875 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
876
877 mData.rollback();
878}
879
880/**
881 * @note Locks this object for writing, together with the peer object (also
882 * for writing) if there is one.
883 */
884void VRDEServer::i_commit()
885{
886 /* sanity */
887 AutoCaller autoCaller(this);
888 AssertComRCReturnVoid(autoCaller.rc());
889
890 /* sanity too */
891 AutoCaller peerCaller(mPeer);
892 AssertComRCReturnVoid(peerCaller.rc());
893
894 /* lock both for writing since we modify both (mPeer is "master" so locked
895 * first) */
896 AutoMultiWriteLock2 alock(mPeer, this COMMA_LOCKVAL_SRC_POS);
897
898 if (mData.isBackedUp())
899 {
900 mData.commit();
901 if (mPeer)
902 {
903 /* attach new data to the peer and reshare it */
904 mPeer->mData.attach(mData);
905 }
906 }
907}
908
909/**
910 * @note Locks this object for writing, together with the peer object
911 * represented by @a aThat (locked for reading).
912 */
913void VRDEServer::i_copyFrom(VRDEServer *aThat)
914{
915 AssertReturnVoid(aThat != NULL);
916
917 /* sanity */
918 AutoCaller autoCaller(this);
919 AssertComRCReturnVoid(autoCaller.rc());
920
921 /* sanity too */
922 AutoCaller thatCaller(aThat);
923 AssertComRCReturnVoid(thatCaller.rc());
924
925 /* peer is not modified, lock it for reading (aThat is "master" so locked
926 * first) */
927 AutoReadLock rl(aThat COMMA_LOCKVAL_SRC_POS);
928 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
929
930 /* this will back up current data */
931 mData.assignCopy(aThat->mData);
932}
933/* vi: set tabstop=4 shiftwidth=4 expandtab: */
Note: See TracBrowser for help on using the repository browser.

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