VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/RecordingSettingsImpl.cpp@ 75455

Last change on this file since 75455 was 75455, checked in by vboxsync, 6 years ago

Recording/Main: Renaming (onRecordChange -> onRecordingChange).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 18.5 KB
Line 
1/* $Id: RecordingSettingsImpl.cpp 75455 2018-11-14 15:04:57Z vboxsync $ */
2/** @file
3 *
4 * VirtualBox COM class implementation - Machine capture settings.
5 */
6
7/*
8 * Copyright (C) 2018 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#define LOG_GROUP LOG_GROUP_MAIN_RECORDINGSETTINGS
20#include "LoggingNew.h"
21
22#include "RecordingSettingsImpl.h"
23#include "RecordingScreenSettingsImpl.h"
24#include "MachineImpl.h"
25
26#include <iprt/cpp/utils.h>
27#include <VBox/settings.h>
28
29#include "AutoStateDep.h"
30#include "AutoCaller.h"
31#include "Global.h"
32
33////////////////////////////////////////////////////////////////////////////////
34//
35// RecordSettings private data definition
36//
37////////////////////////////////////////////////////////////////////////////////
38
39struct RecordingSettings::Data
40{
41 Data()
42 : pMachine(NULL)
43 { }
44
45 Machine * const pMachine;
46 ComObjPtr<RecordingSettings> pPeer;
47 RecordScreenSettingsMap mapScreenObj;
48 bool fHasMachineLock;
49
50 // use the XML settings structure in the members for simplicity
51 Backupable<settings::RecordingSettings> bd;
52};
53
54DEFINE_EMPTY_CTOR_DTOR(RecordingSettings)
55
56HRESULT RecordingSettings::FinalConstruct()
57{
58 return BaseFinalConstruct();
59}
60
61void RecordingSettings::FinalRelease()
62{
63 uninit();
64 BaseFinalRelease();
65}
66
67/**
68 * Initializes the audio adapter object.
69 *
70 * @returns COM result indicator
71 */
72HRESULT RecordingSettings::init(Machine *aParent)
73{
74 LogFlowThisFuncEnter();
75 LogFlowThisFunc(("aParent: %p\n", aParent));
76
77 ComAssertRet(aParent, E_INVALIDARG);
78
79 /* Enclose the state transition NotReady->InInit->Ready */
80 AutoInitSpan autoInitSpan(this);
81 AssertReturn(autoInitSpan.isOk(), E_FAIL);
82
83 m = new Data();
84
85 /* share the parent weakly */
86 unconst(m->pMachine) = aParent;
87
88 m->bd.allocate();
89 m->fHasMachineLock = false;
90
91 autoInitSpan.setSucceeded();
92
93 LogFlowThisFuncLeave();
94 return S_OK;
95}
96
97/**
98 * Initializes the capture settings object given another capture settings object
99 * (a kind of copy constructor). This object shares data with
100 * the object passed as an argument.
101 *
102 * @note This object must be destroyed before the original object
103 * it shares data with is destroyed.
104 */
105HRESULT RecordingSettings::init(Machine *aParent, RecordingSettings *that)
106{
107 LogFlowThisFuncEnter();
108 LogFlowThisFunc(("aParent: %p, that: %p\n", aParent, that));
109
110 ComAssertRet(aParent && that, E_INVALIDARG);
111
112 /* Enclose the state transition NotReady->InInit->Ready */
113 AutoInitSpan autoInitSpan(this);
114 AssertReturn(autoInitSpan.isOk(), E_FAIL);
115
116 m = new Data();
117
118 unconst(m->pMachine) = aParent;
119 m->pPeer = that;
120
121 AutoWriteLock thatlock(that COMMA_LOCKVAL_SRC_POS);
122
123 m->bd.share(that->m->bd);
124 m->mapScreenObj = that->m->mapScreenObj;
125 m->fHasMachineLock = false;
126
127 autoInitSpan.setSucceeded();
128
129 LogFlowThisFuncLeave();
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 */
138HRESULT RecordingSettings::initCopy(Machine *aParent, RecordingSettings *that)
139{
140 LogFlowThisFuncEnter();
141 LogFlowThisFunc(("aParent: %p, that: %p\n", aParent, that));
142
143 ComAssertRet(aParent && that, E_INVALIDARG);
144
145 /* Enclose the state transition NotReady->InInit->Ready */
146 AutoInitSpan autoInitSpan(this);
147 AssertReturn(autoInitSpan.isOk(), E_FAIL);
148
149 m = new Data();
150
151 unconst(m->pMachine) = aParent;
152 // mPeer is left null
153
154 AutoWriteLock thatlock(that COMMA_LOCKVAL_SRC_POS);
155
156 m->bd.attachCopy(that->m->bd);
157 m->mapScreenObj = that->m->mapScreenObj;
158 m->fHasMachineLock = false;
159
160 autoInitSpan.setSucceeded();
161
162 LogFlowThisFuncLeave();
163 return S_OK;
164}
165
166/**
167 * Uninitializes the instance and sets the ready flag to FALSE.
168 * Called either from FinalRelease() or by the parent when it gets destroyed.
169 */
170void RecordingSettings::uninit()
171{
172 LogFlowThisFuncEnter();
173
174 /* Enclose the state transition Ready->InUninit->NotReady */
175 AutoUninitSpan autoUninitSpan(this);
176 if (autoUninitSpan.uninitDone())
177 return;
178
179 /* Note: Do *not* call i_reset() here, as the shared recording configuration
180 * otherwise gets destructed when this object goes out of scope or is destroyed. */
181
182 m->bd.free();
183
184 unconst(m->pPeer) = NULL;
185 unconst(m->pMachine) = NULL;
186
187 delete m;
188 m = NULL;
189
190 LogFlowThisFuncLeave();
191}
192
193// IRecordSettings properties
194/////////////////////////////////////////////////////////////////////////////
195
196HRESULT RecordingSettings::getEnabled(BOOL *enabled)
197{
198 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
199
200 *enabled = m->bd->fEnabled;
201
202 return S_OK;
203}
204
205HRESULT RecordingSettings::setEnabled(BOOL enable)
206{
207 LogFlowThisFuncEnter();
208
209 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
210
211 const bool fEnabled = RT_BOOL(enable);
212
213 HRESULT rc = S_OK;
214
215 if (m->bd->fEnabled != fEnabled)
216 {
217 m->bd.backup();
218 m->bd->fEnabled = fEnabled;
219
220 alock.release();
221 rc = m->pMachine->i_onRecordingChange();
222 if (FAILED(rc))
223 {
224 /*
225 * Normally we would do the actual change _after_ i_onCaptureChange() succeeded.
226 * We cannot do this because that function uses RecordSettings::GetEnabled to
227 * determine if it should start or stop capturing. Therefore we need to manually
228 * undo change.
229 */
230 alock.acquire();
231 m->bd->fEnabled = m->bd.backedUpData()->fEnabled;
232 }
233 else
234 {
235 AutoWriteLock mlock(m->pMachine COMMA_LOCKVAL_SRC_POS); // mParent is const, needs no locking
236 m->pMachine->i_setModified(Machine::IsModified_Recording);
237
238 /* We need to indicate here that we just took the machine lock, as Machine::i_saveSettings() will
239 * call i_commit(), which in turn also wants to lock the machine for writing. */
240 m->fHasMachineLock = true;
241
242 /** Save settings if online - @todo why is this required? -- @bugref{6818} */
243 if (Global::IsOnline(m->pMachine->i_getMachineState()))
244 rc = m->pMachine->i_saveSettings(NULL);
245
246 m->fHasMachineLock = false;
247 }
248 }
249
250 return rc;
251}
252
253HRESULT RecordingSettings::getScreens(std::vector<ComPtr<IRecordingScreenSettings> > &aRecordScreenSettings)
254{
255 LogFlowThisFuncEnter();
256
257 i_syncToMachineDisplays();
258
259 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
260
261 aRecordScreenSettings.clear();
262 aRecordScreenSettings.resize(m->mapScreenObj.size());
263
264 RecordScreenSettingsMap::const_iterator itScreenSettings = m->mapScreenObj.begin();
265 size_t i = 0;
266 while (itScreenSettings != m->mapScreenObj.end())
267 {
268 itScreenSettings->second.queryInterfaceTo(aRecordScreenSettings[i].asOutParam());
269 Assert(aRecordScreenSettings[i].isNotNull());
270 ++i;
271 ++itScreenSettings;
272 }
273
274 Assert(aRecordScreenSettings.size() == m->mapScreenObj.size());
275
276 return S_OK;
277}
278
279HRESULT RecordingSettings::getScreenSettings(ULONG uScreenId, ComPtr<IRecordingScreenSettings> &aRecordScreenSettings)
280{
281 LogFlowThisFuncEnter();
282
283 i_syncToMachineDisplays();
284
285 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
286
287 if (uScreenId + 1 > m->mapScreenObj.size())
288 return setError(E_INVALIDARG, tr("Invalid screen ID specified"));
289
290 RecordScreenSettingsMap::const_iterator itScreenSettings = m->mapScreenObj.find(uScreenId);
291 if (itScreenSettings != m->mapScreenObj.end())
292 {
293 itScreenSettings->second.queryInterfaceTo(aRecordScreenSettings.asOutParam());
294 return S_OK;
295 }
296
297 return VBOX_E_OBJECT_NOT_FOUND;
298}
299
300// IRecordSettings methods
301/////////////////////////////////////////////////////////////////////////////
302
303// public methods only for internal purposes
304/////////////////////////////////////////////////////////////////////////////
305
306/**
307 * Adds a screen settings object to a particular map.
308 *
309 * @returns IPRT status code. VERR_ALREADY_EXISTS if the object in question already exists.
310 * @param screenSettingsMap Map to add screen settings to.
311 * @param uScreenId Screen ID to add settings for.
312 * @param data Recording screen settings to use for that screen.
313 */
314int RecordingSettings::i_createScreenObj(RecordScreenSettingsMap &screenSettingsMap,
315 uint32_t uScreenId, const settings::RecordingScreenSettings &data)
316{
317 LogFlowThisFunc(("Screen %RU32\n", uScreenId));
318
319 if (screenSettingsMap.find(uScreenId) != screenSettingsMap.end())
320 {
321 AssertFailed();
322 return VERR_ALREADY_EXISTS;
323 }
324
325 int vrc = VINF_SUCCESS;
326
327 ComObjPtr<RecordingScreenSettings> recordingScreenSettings;
328 HRESULT rc = recordingScreenSettings.createObject();
329 if (SUCCEEDED(rc))
330 {
331 rc = recordingScreenSettings->init(this, uScreenId, data);
332 if (SUCCEEDED(rc))
333 {
334 try
335 {
336 screenSettingsMap[uScreenId] = recordingScreenSettings;
337 }
338 catch (std::bad_alloc &)
339 {
340 vrc = VERR_NO_MEMORY;
341 }
342 }
343 }
344
345 return vrc;
346}
347
348/**
349 * Removes a screen settings object from a particular map.
350 *
351 * @returns IPRT status code. VERR_NOT_FOUND if specified screen was not found.
352 * @param screenSettingsMap Map to remove screen settings from.
353 * @param uScreenId ID of screen to remove.
354 */
355int RecordingSettings::i_destroyScreenObj(RecordScreenSettingsMap &screenSettingsMap, uint32_t uScreenId)
356{
357 LogFlowThisFunc(("Screen %RU32\n", uScreenId));
358
359 AssertReturn(uScreenId > 0, VERR_INVALID_PARAMETER); /* Removing screen 0 isn't a good idea. */
360
361 RecordScreenSettingsMap::iterator itScreen = screenSettingsMap.find(uScreenId);
362 if (itScreen == screenSettingsMap.end())
363 {
364 AssertFailed();
365 return VERR_NOT_FOUND;
366 }
367
368 /* Make sure to consume the pointer before the one of the
369 * iterator gets released. */
370 ComObjPtr<RecordingScreenSettings> pScreenSettings = itScreen->second;
371
372 screenSettingsMap.erase(itScreen);
373
374 pScreenSettings.setNull();
375
376 return VINF_SUCCESS;
377}
378
379/**
380 * Destroys all screen settings objects of a particular map.
381 *
382 * @returns IPRT status code.
383 * @param screenSettingsMap Map to destroy screen settings objects for.
384 */
385int RecordingSettings::i_destroyAllScreenObj(RecordScreenSettingsMap &screenSettingsMap)
386{
387 LogFlowThisFuncEnter();
388
389 RecordScreenSettingsMap::iterator itScreen = screenSettingsMap.begin();
390 if (itScreen != screenSettingsMap.end())
391 {
392 /* Make sure to consume the pointer before the one of the
393 * iterator gets released. */
394 ComObjPtr<RecordingScreenSettings> pScreenSettings = itScreen->second;
395
396 screenSettingsMap.erase(itScreen);
397
398 pScreenSettings.setNull();
399
400 itScreen = screenSettingsMap.begin();
401 }
402
403 return VINF_SUCCESS;
404}
405
406/**
407 * Loads settings from the given settings.
408 * May be called once right after this object creation.
409 *
410 * @param data Capture settings to load from.
411 *
412 * @note Locks this object for writing.
413 */
414HRESULT RecordingSettings::i_loadSettings(const settings::RecordingSettings &data)
415{
416 LogFlowThisFuncEnter();
417
418 AutoCaller autoCaller(this);
419 AssertComRCReturnRC(autoCaller.rc());
420
421 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
422
423 HRESULT rc = S_OK;
424
425 i_reset();
426
427 LogFlowThisFunc(("Data has %zu screens\n", data.mapScreens.size()));
428
429 settings::RecordingScreenMap::const_iterator itScreen = data.mapScreens.begin();
430 while (itScreen != data.mapScreens.end())
431 {
432 int vrc = i_createScreenObj(m->mapScreenObj,
433 itScreen->first /* uScreenId */, itScreen->second /* Settings */);
434 if (RT_FAILURE(vrc))
435 {
436 rc = E_OUTOFMEMORY;
437 break;
438 }
439
440 ++itScreen;
441 }
442
443 if (FAILED(rc))
444 return rc;
445
446 ComAssertComRC(rc);
447 Assert(m->mapScreenObj.size() == data.mapScreens.size());
448
449 // simply copy
450 m->bd.assignCopy(&data);
451
452 LogFlowThisFunc(("Returning %Rhrc\n", rc));
453 return rc;
454}
455
456/**
457 * Resets the internal object state by destroying all screen settings objects.
458 */
459void RecordingSettings::i_reset(void)
460{
461 LogFlowThisFuncEnter();
462
463 i_destroyAllScreenObj(m->mapScreenObj);
464 m->bd->mapScreens.clear();
465}
466
467/**
468 * Saves settings to the given settings.
469 *
470 * @param data Where to store the capture settings to.
471 *
472 * @note Locks this object for reading.
473 */
474HRESULT RecordingSettings::i_saveSettings(settings::RecordingSettings &data)
475{
476 LogFlowThisFuncEnter();
477
478 AutoCaller autoCaller(this);
479 AssertComRCReturnRC(autoCaller.rc());
480
481 int rc2 = i_syncToMachineDisplays();
482 AssertRC(rc2);
483
484 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
485
486 data = *m->bd.data();
487
488 settings::RecordingScreenMap::iterator itScreen = data.mapScreens.begin();
489 while (itScreen != data.mapScreens.end())
490 {
491 /* Store relative path of capture file if possible. */
492 m->pMachine->i_copyPathRelativeToMachine(itScreen->second.File.strName /* Source */,
493 itScreen->second.File.strName /* Target */);
494 ++itScreen;
495 }
496
497 LogFlowThisFuncLeave();
498 return S_OK;
499}
500
501void RecordingSettings::i_rollback()
502{
503 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
504 m->bd.rollback();
505}
506
507void RecordingSettings::i_commit()
508{
509 /* sanity */
510 AutoCaller autoCaller(this);
511 AssertComRCReturnVoid(autoCaller.rc());
512
513 /* sanity too */
514 AutoCaller peerCaller(m->pPeer);
515 AssertComRCReturnVoid(peerCaller.rc());
516
517 /* lock both for writing since we modify both (mPeer is "master" so locked
518 * first) */
519 AutoMultiWriteLock2 alock(m->pPeer, this COMMA_LOCKVAL_SRC_POS);
520
521 if (m->bd.isBackedUp())
522 {
523 m->bd.commit();
524 if (m->pPeer)
525 {
526 /* attach new data to the peer and reshare it */
527 AutoWriteLock peerlock(m->pPeer COMMA_LOCKVAL_SRC_POS);
528 m->pPeer->m->bd.attach(m->bd);
529 }
530 }
531}
532
533void RecordingSettings::i_copyFrom(RecordingSettings *aThat)
534{
535 AssertReturnVoid(aThat != NULL);
536
537 /* sanity */
538 AutoCaller autoCaller(this);
539 AssertComRCReturnVoid(autoCaller.rc());
540
541 /* sanity too */
542 AutoCaller thatCaller(aThat);
543 AssertComRCReturnVoid(thatCaller.rc());
544
545 /* peer is not modified, lock it for reading (aThat is "master" so locked
546 * first) */
547 AutoReadLock rl(aThat COMMA_LOCKVAL_SRC_POS);
548 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
549
550 /* this will back up current data */
551 m->bd.assignCopy(aThat->m->bd);
552}
553
554void RecordingSettings::i_applyDefaults(void)
555{
556 /* sanity */
557 AutoCaller autoCaller(this);
558 AssertComRCReturnVoid(autoCaller.rc());
559
560 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
561
562 /* Initialize default capturing settings here. */
563}
564
565/**
566 * Returns the full path to the default video capture file.
567 */
568int RecordingSettings::i_getDefaultFileName(Utf8Str &strFile, bool fWithFileExtension)
569{
570 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
571
572 strFile = m->pMachine->i_getSettingsFileFull(); // path/to/machinesfolder/vmname/vmname.vbox
573 strFile.stripSuffix(); // path/to/machinesfolder/vmname/vmname
574 if (fWithFileExtension)
575 strFile.append(".webm"); // path/to/machinesfolder/vmname/vmname.webm
576
577 return VINF_SUCCESS;
578}
579
580/**
581 * Determines whether the recording settings currently can be changed or not.
582 *
583 * @returns \c true if the settings can be changed, \c false if not.
584 */
585bool RecordingSettings::i_canChangeSettings(void)
586{
587 AutoAnyStateDependency adep(m->pMachine);
588 if (FAILED(adep.rc()))
589 return false;
590
591 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
592
593 /* Only allow settings to be changed when recording is disabled when the machine is running. */
594 if ( Global::IsOnline(adep.machineState())
595 && m->bd->fEnabled)
596 {
597 return false;
598 }
599
600 return true;
601}
602
603/**
604 * Gets called when the machine object needs to know that the recording settings
605 * have been changed.
606 */
607void RecordingSettings::i_onSettingsChanged(void)
608{
609 LogFlowThisFuncEnter();
610
611 AutoWriteLock mlock(m->pMachine COMMA_LOCKVAL_SRC_POS);
612 m->pMachine->i_setModified(Machine::IsModified_Recording);
613 mlock.release();
614
615 LogFlowThisFuncLeave();
616}
617
618/**
619 * Synchronizes the screen settings (COM) objects and configuration data
620 * to the number of the machine's configured displays.
621 */
622int RecordingSettings::i_syncToMachineDisplays(void)
623{
624 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
625
626 AssertPtr(m->pMachine);
627 const ULONG cMonitors = m->pMachine->i_getMonitorCount();
628
629 LogFlowThisFunc(("cMonitors=%RU32\n", cMonitors));
630 LogFlowThisFunc(("Data screen count = %zu, COM object count = %zu\n", m->bd->mapScreens.size(), m->mapScreenObj.size()));
631
632 /* If counts match, take a shortcut. */
633 if (cMonitors == m->mapScreenObj.size())
634 return VINF_SUCCESS;
635
636 /* Create all new screen settings objects which are not there yet. */
637 for (ULONG i = 0; i < cMonitors; i++)
638 {
639 if (m->mapScreenObj.find(i) == m->mapScreenObj.end())
640 {
641 settings::RecordingScreenMap::const_iterator itScreen = m->bd->mapScreens.find(i);
642 if (itScreen == m->bd->mapScreens.end())
643 {
644 settings::RecordingScreenSettings defaultScreenSettings; /* Apply default settings. */
645 m->bd->mapScreens[i] = defaultScreenSettings;
646 }
647
648 int vrc2 = i_createScreenObj(m->mapScreenObj, i /* Screen ID */, m->bd->mapScreens[i]);
649 AssertRC(vrc2);
650 }
651 }
652
653 /* Remove all left over screen settings objects which are not needed anymore. */
654 const ULONG cSettings = (ULONG)m->mapScreenObj.size();
655 for (ULONG i = cMonitors; i < cSettings; i++)
656 {
657 m->bd->mapScreens.erase(i);
658 int vrc2 = i_destroyScreenObj(m->mapScreenObj, i /* Screen ID */);
659 AssertRC(vrc2);
660 }
661
662 Assert(m->mapScreenObj.size() == cMonitors);
663 Assert(m->bd->mapScreens.size() == cMonitors);
664
665 LogFlowThisFuncLeave();
666 return VINF_SUCCESS;
667}
668
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