VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/BandwidthGroupImpl.cpp@ 49953

Last change on this file since 49953 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: 8.7 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 "BandwidthGroupImpl.h"
19#include "MachineImpl.h"
20#include "Global.h"
21
22#include "AutoCaller.h"
23#include "Logging.h"
24
25#include <iprt/cpp/utils.h>
26
27// constructor / destructor
28/////////////////////////////////////////////////////////////////////////////
29//
30DEFINE_EMPTY_CTOR_DTOR(BandwidthGroup)
31
32HRESULT BandwidthGroup::FinalConstruct()
33{
34 return BaseFinalConstruct();
35}
36
37void BandwidthGroup::FinalRelease()
38{
39 uninit();
40 BaseFinalRelease();
41}
42
43// public initializer/uninitializer for internal purposes only
44/////////////////////////////////////////////////////////////////////////////
45
46/**
47 * Initializes the bandwidth group object.
48 *
49 * @returns COM result indicator.
50 * @param aParent Pointer to our parent object.
51 * @param aName Name of the storage controller.
52 * @param aInstance Instance number of the storage controller.
53 */
54HRESULT BandwidthGroup::init(BandwidthControl *aParent,
55 const Utf8Str &aName,
56 BandwidthGroupType_T aType,
57 LONG64 aMaxBytesPerSec)
58{
59 LogFlowThisFunc(("aParent=%p aName=\"%s\"\n",
60 aParent, aName.c_str()));
61
62 ComAssertRet(aParent && !aName.isEmpty(), E_INVALIDARG);
63 if ( (aType <= BandwidthGroupType_Null)
64 || (aType > BandwidthGroupType_Network))
65 return setError(E_INVALIDARG,
66 tr("Invalid bandwidth group type type"));
67
68 /* Enclose the state transition NotReady->InInit->Ready */
69 AutoInitSpan autoInitSpan(this);
70 AssertReturn(autoInitSpan.isOk(), E_FAIL);
71
72 m = new Data(aParent);
73
74 /* m->pPeer is left null */
75
76 m->bd.allocate();
77
78 m->bd->strName = aName;
79 m->bd->enmType = aType;
80 m->bd->cReferences = 0;
81 m->bd->aMaxBytesPerSec = aMaxBytesPerSec;
82
83 /* Confirm a successful initialization */
84 autoInitSpan.setSucceeded();
85
86 return S_OK;
87}
88
89/**
90 * Initializes the object given another object
91 * (a kind of copy constructor). This object shares data with
92 * the object passed as an argument.
93 *
94 * @param aReshare
95 * When false, the original object will remain a data owner.
96 * Otherwise, data ownership will be transferred from the original
97 * object to this one.
98 *
99 * @note This object must be destroyed before the original object
100 * it shares data with is destroyed.
101 *
102 * @note Locks @a aThat object for writing if @a aReshare is @c true, or for
103 * reading if @a aReshare is false.
104 */
105HRESULT BandwidthGroup::init(BandwidthControl *aParent,
106 BandwidthGroup *aThat,
107 bool aReshare /* = false */)
108{
109 LogFlowThisFunc(("aParent=%p, aThat=%p, aReshare=%RTbool\n",
110 aParent, aThat, aReshare));
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 m = new Data(aParent);
119
120 /* sanity */
121 AutoCaller thatCaller(aThat);
122 AssertComRCReturnRC(thatCaller.rc());
123
124 if (aReshare)
125 {
126 AutoWriteLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
127
128 unconst(aThat->m->pPeer) = this;
129 m->bd.attach(aThat->m->bd);
130 }
131 else
132 {
133 unconst(m->pPeer) = aThat;
134
135 AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
136 m->bd.share(aThat->m->bd);
137 }
138
139 /* Confirm successful initialization */
140 autoInitSpan.setSucceeded();
141
142 return S_OK;
143}
144
145/**
146 * Initializes the storage controller object given another guest object
147 * (a kind of copy constructor). This object makes a private copy of data
148 * of the original object passed as an argument.
149 */
150HRESULT BandwidthGroup::initCopy(BandwidthControl *aParent, BandwidthGroup *aThat)
151{
152 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
153
154 ComAssertRet(aParent && aThat, E_INVALIDARG);
155
156 /* Enclose the state transition NotReady->InInit->Ready */
157 AutoInitSpan autoInitSpan(this);
158 AssertReturn(autoInitSpan.isOk(), E_FAIL);
159
160 m = new Data(aParent);
161 /* m->pPeer is left null */
162
163 AutoCaller thatCaller(aThat);
164 AssertComRCReturnRC(thatCaller.rc());
165
166 AutoReadLock thatlock(aThat COMMA_LOCKVAL_SRC_POS);
167 m->bd.attachCopy(aThat->m->bd);
168
169 /* Confirm a successful initialization */
170 autoInitSpan.setSucceeded();
171
172 return S_OK;
173}
174
175
176/**
177 * Uninitializes the instance and sets the ready flag to FALSE.
178 * Called either from FinalRelease() or by the parent when it gets destroyed.
179 */
180void BandwidthGroup::uninit()
181{
182 LogFlowThisFunc(("\n"));
183
184 /* Enclose the state transition Ready->InUninit->NotReady */
185 AutoUninitSpan autoUninitSpan(this);
186 if (autoUninitSpan.uninitDone())
187 return;
188
189 m->bd.free();
190
191 unconst(m->pPeer) = NULL;
192 unconst(m->pParent) = NULL;
193
194 delete m;
195 m = NULL;
196}
197
198HRESULT BandwidthGroup::getName(com::Utf8Str &aName)
199{
200 /* mName is constant during life time, no need to lock */
201 aName = m->bd.data()->strName;
202
203 return S_OK;
204}
205
206HRESULT BandwidthGroup::getType(BandwidthGroupType_T *aType)
207{
208 /* type is constant during life time, no need to lock */
209 *aType = m->bd->enmType;
210
211 return S_OK;
212}
213
214HRESULT BandwidthGroup::getReference(ULONG *aReferences)
215{
216 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
217
218 *aReferences = m->bd->cReferences;
219
220 return S_OK;
221}
222
223HRESULT BandwidthGroup::getMaxBytesPerSec(LONG64 *aMaxBytesPerSec)
224{
225 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
226
227 *aMaxBytesPerSec = m->bd->aMaxBytesPerSec;
228
229 return S_OK;
230}
231
232HRESULT BandwidthGroup::setMaxBytesPerSec(LONG64 aMaxBytesPerSec)
233{
234 if (aMaxBytesPerSec < 0)
235 return setError(E_INVALIDARG,
236 tr("Bandwidth group limit cannot be negative"));
237
238 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
239
240 m->bd.backup();
241 m->bd->aMaxBytesPerSec = aMaxBytesPerSec;
242
243 /* inform direct session if any. */
244 ComObjPtr<Machine> pMachine = m->pParent->i_getMachine();
245 alock.release();
246 pMachine->onBandwidthGroupChange(this);
247
248 return S_OK;
249}
250
251// public methods only for internal purposes
252/////////////////////////////////////////////////////////////////////////////
253
254/** @note Locks objects for writing! */
255void BandwidthGroup::i_rollback()
256{
257 AutoCaller autoCaller(this);
258 AssertComRCReturnVoid(autoCaller.rc());
259
260 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
261
262 m->bd.rollback();
263}
264
265/**
266 * @note Locks this object for writing, together with the peer object (also
267 * for writing) if there is one.
268 */
269void BandwidthGroup::i_commit()
270{
271 /* sanity */
272 AutoCaller autoCaller(this);
273 AssertComRCReturnVoid(autoCaller.rc());
274
275 /* sanity too */
276 AutoCaller peerCaller(m->pPeer);
277 AssertComRCReturnVoid(peerCaller.rc());
278
279 /* lock both for writing since we modify both (m->pPeer is "master" so locked
280 * first) */
281 AutoMultiWriteLock2 alock(m->pPeer, this COMMA_LOCKVAL_SRC_POS);
282
283 if (m->bd.isBackedUp())
284 {
285 m->bd.commit();
286 if (m->pPeer)
287 {
288 // attach new data to the peer and reshare it
289 m->pPeer->m->bd.attach(m->bd);
290 }
291 }
292}
293
294
295/**
296 * Cancels sharing (if any) by making an independent copy of data.
297 * This operation also resets this object's peer to NULL.
298 *
299 * @note Locks this object for writing, together with the peer object
300 * represented by @a aThat (locked for reading).
301 */
302void BandwidthGroup::i_unshare()
303{
304 /* sanity */
305 AutoCaller autoCaller(this);
306 AssertComRCReturnVoid(autoCaller.rc());
307
308 /* sanity too */
309 AutoCaller peerCaller(m->pPeer);
310 AssertComRCReturnVoid(peerCaller.rc());
311
312 /* peer is not modified, lock it for reading (m->pPeer is "master" so locked
313 * first) */
314 AutoReadLock rl(m->pPeer COMMA_LOCKVAL_SRC_POS);
315 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
316
317 if (m->bd.isShared())
318 {
319 if (!m->bd.isBackedUp())
320 m->bd.backup();
321
322 m->bd.commit();
323 }
324
325 unconst(m->pPeer) = NULL;
326}
327
328void BandwidthGroup::i_reference()
329{
330 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
331 m->bd.backup();
332 m->bd->cReferences++;
333}
334
335void BandwidthGroup::i_release()
336{
337 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
338 m->bd.backup();
339 m->bd->cReferences--;
340}
341
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