VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/BandwidthControlImpl.cpp@ 49963

Last change on this file since 49963 was 49738, checked in by vboxsync, 11 years ago

Main/BandwidthControl: fix regression (iterator incremented twice), plus lots of whitespace cleanup

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.8 KB
Line 
1/** @file
2 *
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2006-2013 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 "BandwidthControlImpl.h"
19#include "BandwidthGroupImpl.h"
20#include "MachineImpl.h"
21#include "Global.h"
22
23#include "AutoStateDep.h"
24#include "AutoCaller.h"
25#include "Logging.h"
26
27#include <iprt/cpp/utils.h>
28#include <VBox/com/array.h>
29#include <algorithm>
30
31// defines
32/////////////////////////////////////////////////////////////////////////////
33
34// constructor / destructor
35/////////////////////////////////////////////////////////////////////////////
36DEFINE_EMPTY_CTOR_DTOR(BandwidthControl)
37
38
39HRESULT BandwidthControl::FinalConstruct()
40{
41 return BaseFinalConstruct();
42}
43
44void BandwidthControl::FinalRelease()
45{
46 uninit();
47 BaseFinalRelease();
48}
49
50// public initializer/uninitializer for internal purposes only
51/////////////////////////////////////////////////////////////////////////////
52
53/**
54 * Initializes the bandwidth group object.
55 *
56 * @returns COM result indicator.
57 * @param aParent Pointer to our parent object.
58 * @param aName Name of the storage controller.
59 * @param aInstance Instance number of the storage controller.
60 */
61HRESULT BandwidthControl::init(Machine *aParent)
62{
63 LogFlowThisFunc(("aParent=%p\n", aParent));
64
65 ComAssertRet(aParent, E_INVALIDARG);
66
67 /* Enclose the state transition NotReady->InInit->Ready */
68 AutoInitSpan autoInitSpan(this);
69 AssertReturn(autoInitSpan.isOk(), E_FAIL);
70
71 m = new Data(aParent);
72
73 /* m->pPeer is left null */
74
75 m->llBandwidthGroups.allocate();
76
77 /* Confirm a successful initialization */
78 autoInitSpan.setSucceeded();
79
80 return S_OK;
81}
82
83/**
84 * Initializes the object given another object
85 * (a kind of copy constructor). This object shares data with
86 * the object passed as an argument.
87 *
88 * @note This object must be destroyed before the original object
89 * it shares data with is destroyed.
90 *
91 * @note Locks @a aThat object for writing if @a aReshare is @c true, or for
92 * reading if @a aReshare is false.
93 */
94HRESULT BandwidthControl::init(Machine *aParent,
95 BandwidthControl *aThat)
96{
97 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
98
99 ComAssertRet(aParent && aThat, E_INVALIDARG);
100
101 /* Enclose the state transition NotReady->InInit->Ready */
102 AutoInitSpan autoInitSpan(this);
103 AssertReturn(autoInitSpan.isOk(), E_FAIL);
104
105 m = new Data(aParent);
106
107 /* sanity */
108 AutoCaller thatCaller(aThat);
109 AssertComRCReturnRC(thatCaller.rc());
110
111 unconst(m->pPeer) = aThat;
112 AutoWriteLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
113
114 /* create copies of all groups */
115 m->llBandwidthGroups.allocate();
116 BandwidthGroupList::const_iterator it;
117 for (it = aThat->m->llBandwidthGroups->begin();
118 it != aThat->m->llBandwidthGroups->end();
119 ++it)
120 {
121 ComObjPtr<BandwidthGroup> group;
122 group.createObject();
123 group->init(this, *it);
124 m->llBandwidthGroups->push_back(group);
125 }
126
127 /* Confirm successful initialization */
128 autoInitSpan.setSucceeded();
129
130 return S_OK;
131}
132
133/**
134 * Initializes the storage controller 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 BandwidthControl::initCopy(Machine *aParent, BandwidthControl *aThat)
139{
140 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
141
142 ComAssertRet(aParent && aThat, E_INVALIDARG);
143
144 /* Enclose the state transition NotReady->InInit->Ready */
145 AutoInitSpan autoInitSpan(this);
146 AssertReturn(autoInitSpan.isOk(), E_FAIL);
147
148 m = new Data(aParent);
149 /* m->pPeer is left null */
150
151 AutoCaller thatCaller(aThat);
152 AssertComRCReturnRC(thatCaller.rc());
153
154 AutoReadLock thatlock(aThat COMMA_LOCKVAL_SRC_POS);
155
156 /* create copies of all groups */
157 m->llBandwidthGroups.allocate();
158 BandwidthGroupList::const_iterator it;
159 for (it = aThat->m->llBandwidthGroups->begin();
160 it != aThat->m->llBandwidthGroups->end();
161 ++it)
162 {
163 ComObjPtr<BandwidthGroup> group;
164 group.createObject();
165 group->init(this, *it);
166 m->llBandwidthGroups->push_back(group);
167 }
168
169 /* Confirm a successful initialization */
170 autoInitSpan.setSucceeded();
171
172 return S_OK;
173}
174
175
176/**
177 * @note Locks this object for writing, together with the peer object
178 * represented by @a aThat (locked for reading).
179 */
180void BandwidthControl::i_copyFrom(BandwidthControl *aThat)
181{
182 AssertReturnVoid(aThat != NULL);
183
184 /* sanity */
185 AutoCaller autoCaller(this);
186 AssertComRCReturnVoid(autoCaller.rc());
187
188 /* sanity too */
189 AutoCaller thatCaller(aThat);
190 AssertComRCReturnVoid(thatCaller.rc());
191
192 /* even more sanity */
193 AutoAnyStateDependency adep(m->pParent);
194 AssertComRCReturnVoid(adep.rc());
195 /* Machine::copyFrom() may not be called when the VM is running */
196 AssertReturnVoid(!Global::IsOnline(adep.machineState()));
197
198 /* peer is not modified, lock it for reading (aThat is "master" so locked
199 * first) */
200 AutoReadLock rl(aThat COMMA_LOCKVAL_SRC_POS);
201 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
202
203 /* create private copies of all filters */
204 m->llBandwidthGroups.backup();
205 m->llBandwidthGroups->clear();
206 BandwidthGroupList::const_iterator it;
207 for (it = aThat->m->llBandwidthGroups->begin();
208 it != aThat->m->llBandwidthGroups->end();
209 ++it)
210 {
211 ComObjPtr<BandwidthGroup> group;
212 group.createObject();
213 group->initCopy(this, *it);
214 m->llBandwidthGroups->push_back(group);
215 }
216}
217
218/** @note Locks objects for writing! */
219void BandwidthControl::i_rollback()
220{
221 AutoCaller autoCaller(this);
222 AssertComRCReturnVoid(autoCaller.rc());
223
224 /* we need the machine state */
225 AutoAnyStateDependency adep(m->pParent);
226 AssertComRCReturnVoid(adep.rc());
227
228 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
229 BandwidthGroupList::const_iterator it;
230
231 if (!m->llBandwidthGroups.isNull())
232 {
233 if (m->llBandwidthGroups.isBackedUp())
234 {
235 /* unitialize all new groups (absent in the backed up list). */
236 BandwidthGroupList *backedList = m->llBandwidthGroups.backedUpData();
237 for (it = m->llBandwidthGroups->begin();
238 it != m->llBandwidthGroups->end();
239 ++it)
240 {
241 if ( std::find(backedList->begin(), backedList->end(), *it)
242 == backedList->end())
243 (*it)->uninit();
244 }
245
246 /* restore the list */
247 m->llBandwidthGroups.rollback();
248 }
249
250 /* rollback any changes to groups after restoring the list */
251 for (it = m->llBandwidthGroups->begin();
252 it != m->llBandwidthGroups->end();
253 ++it)
254 (*it)->i_rollback();
255 }
256}
257
258void BandwidthControl::i_commit()
259{
260 bool commitBandwidthGroups = false;
261 BandwidthGroupList::const_iterator it;
262
263 if (m->llBandwidthGroups.isBackedUp())
264 {
265 m->llBandwidthGroups.commit();
266
267 if (m->pPeer)
268 {
269 AutoWriteLock peerlock(m->pPeer COMMA_LOCKVAL_SRC_POS);
270
271 /* Commit all changes to new controllers (this will reshare data with
272 * peers for those who have peers) */
273 BandwidthGroupList *newList = new BandwidthGroupList();
274 for (it = m->llBandwidthGroups->begin();
275 it != m->llBandwidthGroups->end();
276 ++it)
277 {
278 (*it)->i_commit();
279
280 /* look if this group has a peer group */
281 ComObjPtr<BandwidthGroup> peer = (*it)->i_getPeer();
282 if (!peer)
283 {
284 /* no peer means the device is a newly created one;
285 * create a peer owning data this device share it with */
286 peer.createObject();
287 peer->init(m->pPeer, *it, true /* aReshare */);
288 }
289 else
290 {
291 /* remove peer from the old list */
292 m->pPeer->m->llBandwidthGroups->remove(peer);
293 }
294 /* and add it to the new list */
295 newList->push_back(peer);
296 }
297
298 /* uninit old peer's controllers that are left */
299 for (it = m->pPeer->m->llBandwidthGroups->begin();
300 it != m->pPeer->m->llBandwidthGroups->end();
301 ++it)
302 (*it)->uninit();
303
304 /* attach new list of controllers to our peer */
305 m->pPeer->m->llBandwidthGroups.attach(newList);
306 }
307 else
308 {
309 /* we have no peer (our parent is the newly created machine);
310 * just commit changes to devices */
311 commitBandwidthGroups = true;
312 }
313 }
314 else
315 {
316 /* the list of groups itself is not changed,
317 * just commit changes to controllers themselves */
318 commitBandwidthGroups = true;
319 }
320
321 if (commitBandwidthGroups)
322 {
323 for (it = m->llBandwidthGroups->begin();
324 it != m->llBandwidthGroups->end();
325 ++it)
326 (*it)->i_commit();
327 }
328}
329
330/**
331 * Uninitializes the instance and sets the ready flag to FALSE.
332 * Called either from FinalRelease() or by the parent when it gets destroyed.
333 */
334void BandwidthControl::uninit()
335{
336 LogFlowThisFunc(("\n"));
337
338 /* Enclose the state transition Ready->InUninit->NotReady */
339 AutoUninitSpan autoUninitSpan(this);
340 if (autoUninitSpan.uninitDone())
341 return;
342
343 // uninit all groups on the list (it's a standard std::list not an ObjectsList
344 // so we must uninit() manually)
345 BandwidthGroupList::iterator it;
346 for (it = m->llBandwidthGroups->begin();
347 it != m->llBandwidthGroups->end();
348 ++it)
349 (*it)->uninit();
350
351 m->llBandwidthGroups.free();
352
353 unconst(m->pPeer) = NULL;
354 unconst(m->pParent) = NULL;
355
356 delete m;
357 m = NULL;
358}
359
360/**
361 * Returns a storage controller object with the given name.
362 *
363 * @param aName storage controller name to find
364 * @param aBandwidthGroup where to return the found storage controller
365 * @param aSetError true to set extended error info on failure
366 */
367HRESULT BandwidthControl::i_getBandwidthGroupByName(const com::Utf8Str &aName,
368 ComObjPtr<BandwidthGroup> &aBandwidthGroup,
369 bool aSetError /* = false */)
370{
371 AssertReturn(!aName.isEmpty(), E_INVALIDARG);
372
373 for (BandwidthGroupList::const_iterator it = m->llBandwidthGroups->begin();
374 it != m->llBandwidthGroups->end();
375 ++it)
376 {
377 if ((*it)->i_getName() == aName)
378 {
379 aBandwidthGroup = (*it);
380 return S_OK;
381 }
382 }
383
384 if (aSetError)
385 return setError(VBOX_E_OBJECT_NOT_FOUND,
386 tr("Could not find a bandwidth group named '%s'"),
387 aName.c_str());
388 return VBOX_E_OBJECT_NOT_FOUND;
389}
390// To do
391HRESULT BandwidthControl::createBandwidthGroup(const com::Utf8Str &aName,
392 BandwidthGroupType_T aType,
393 LONG64 aMaxBytesPerSec)
394{
395 if (aMaxBytesPerSec < 0)
396 return setError(E_INVALIDARG,
397 tr("Bandwidth group limit cannot be negative"));
398
399 /* the machine needs to be mutable */
400 AutoMutableStateDependency adep(m->pParent);
401 if (FAILED(adep.rc())) return adep.rc();
402
403 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
404
405 /* try to find one with the name first. */
406 ComObjPtr<BandwidthGroup> group;
407 HRESULT rc = i_getBandwidthGroupByName(aName, group, false /* aSetError */);
408
409 if (SUCCEEDED(rc))
410 return setError(VBOX_E_OBJECT_IN_USE,
411 tr("Bandwidth group named '%ls' already exists"),
412 Bstr(aName).raw());
413
414 group.createObject();
415
416 rc = group->init(this, aName, aType, aMaxBytesPerSec);
417 if (FAILED(rc)) return rc;
418
419 m->pParent->setModified(Machine::IsModified_BandwidthControl);
420 m->llBandwidthGroups.backup();
421 m->llBandwidthGroups->push_back(group);
422
423 return S_OK;
424}
425
426HRESULT BandwidthControl::deleteBandwidthGroup(const com::Utf8Str &aName)
427{
428 /* the machine needs to be mutable */
429 AutoMutableStateDependency adep(m->pParent);
430 if (FAILED(adep.rc())) return adep.rc();
431
432 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
433
434 ComObjPtr<BandwidthGroup> group;
435 HRESULT rc = i_getBandwidthGroupByName(aName, group, true /* aSetError */);
436 if (FAILED(rc)) return rc;
437
438 if (group->i_getReferences() != 0)
439 return setError(VBOX_E_OBJECT_IN_USE,
440 tr("The bandwidth group '%ls' is still in use"), Bstr(aName).raw());
441
442 /* We can remove it now. */
443 m->pParent->setModified(Machine::IsModified_BandwidthControl);
444 m->llBandwidthGroups.backup();
445
446 group->i_unshare();
447
448 m->llBandwidthGroups->remove(group);
449
450 /* inform the direct session if any */
451 alock.release();
452 //onStorageControllerChange(); @todo
453
454 return S_OK;
455}
456
457HRESULT BandwidthControl::getNumGroups(ULONG *aNumGroups)
458{
459 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
460
461 *aNumGroups = m->llBandwidthGroups->size();
462
463 return S_OK;
464}
465
466HRESULT BandwidthControl::getBandwidthGroup(const com::Utf8Str &aName, ComPtr<IBandwidthGroup> &aBandwidthGroup)
467{
468 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
469
470 ComObjPtr<BandwidthGroup> group;
471 HRESULT rc = i_getBandwidthGroupByName(aName, group, true /* aSetError */);
472
473 if (SUCCEEDED(rc))
474 group.queryInterfaceTo(aBandwidthGroup.asOutParam());
475
476 return rc;
477}
478
479HRESULT BandwidthControl::getAllBandwidthGroups(std::vector<ComPtr<IBandwidthGroup> > &aBandwidthGroups)
480{
481 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
482 aBandwidthGroups.resize(0);
483 BandwidthGroupList::const_iterator it;
484 for (it = m->llBandwidthGroups->begin();
485 it != m->llBandwidthGroups->end();
486 ++it)
487 aBandwidthGroups.push_back(*it);
488
489 return S_OK;
490}
491
492HRESULT BandwidthControl::i_loadSettings(const settings::IOSettings &data)
493{
494 HRESULT rc = S_OK;
495
496 AutoCaller autoCaller(this);
497 AssertComRCReturnRC(autoCaller.rc());
498 settings::BandwidthGroupList::const_iterator it;
499 for (it = data.llBandwidthGroups.begin();
500 it != data.llBandwidthGroups.end();
501 ++it)
502 {
503 const settings::BandwidthGroup &gr = *it;
504 rc = createBandwidthGroup(gr.strName, gr.enmType, gr.cMaxBytesPerSec);
505 if (FAILED(rc)) break;
506 }
507
508 return rc;
509}
510
511HRESULT BandwidthControl::i_saveSettings(settings::IOSettings &data)
512{
513 AutoCaller autoCaller(this);
514 if (FAILED(autoCaller.rc())) return autoCaller.rc();
515
516 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
517 data.llBandwidthGroups.clear();
518 BandwidthGroupList::const_iterator it;
519 for (it = m->llBandwidthGroups->begin();
520 it != m->llBandwidthGroups->end();
521 ++it)
522 {
523 AutoWriteLock groupLock(*it COMMA_LOCKVAL_SRC_POS);
524 settings::BandwidthGroup group;
525
526 group.strName = (*it)->i_getName();
527 group.enmType = (*it)->i_getType();
528 group.cMaxBytesPerSec = (*it)->i_getMaxBytesPerSec();
529
530 data.llBandwidthGroups.push_back(group);
531 }
532
533 return S_OK;
534}
535
536Machine * BandwidthControl::i_getMachine() const
537{
538 return m->pParent;
539}
540
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