VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxManage/VBoxManageBandwidthControl.cpp@ 52320

Last change on this file since 52320 was 50085, checked in by vboxsync, 11 years ago

VBoxManage/BandwidthGroups: additional check

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 11.1 KB
Line 
1/* $Id: VBoxManageBandwidthControl.cpp 50085 2014-01-16 09:42:33Z vboxsync $ */
2/** @file
3 * VBoxManage - The bandwidth control related commands.
4 */
5
6/*
7 * Copyright (C) 2006-2012 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#ifndef VBOX_ONLY_DOCS
19
20/*******************************************************************************
21* Header Files *
22*******************************************************************************/
23#include <VBox/com/com.h>
24#include <VBox/com/array.h>
25#include <VBox/com/ErrorInfo.h>
26#include <VBox/com/errorprint.h>
27#include <VBox/com/VirtualBox.h>
28
29#include <iprt/path.h>
30#include <iprt/param.h>
31#include <iprt/string.h>
32#include <iprt/ctype.h>
33#include <iprt/stream.h>
34#include <iprt/getopt.h>
35#include <VBox/log.h>
36
37#include "VBoxManage.h"
38using namespace com;
39
40
41// funcs
42///////////////////////////////////////////////////////////////////////////////
43
44
45/**
46 * Parses a string in the following format "n[k|m|g|K|M|G]". Stores the value
47 * of n expressed in bytes to *pLimit. k meas kilobit, while K means kilobyte.
48 *
49 * @returns Error message or NULL if successful.
50 * @param pcszLimit The string to parse.
51 * @param pLimit Where to store the result.
52 */
53static const char *parseLimit(const char *pcszLimit, int64_t *pLimit)
54{
55 int iMultiplier = _1M;
56 char *pszNext = NULL;
57 int rc = RTStrToInt64Ex(pcszLimit, &pszNext, 10, pLimit);
58
59 switch (rc)
60 {
61 case VINF_SUCCESS:
62 break;
63 case VWRN_NUMBER_TOO_BIG:
64 return "Limit is too big\n";
65 case VWRN_TRAILING_CHARS:
66 switch (*pszNext)
67 {
68 case 'G': iMultiplier = _1G; break;
69 case 'M': iMultiplier = _1M; break;
70 case 'K': iMultiplier = _1K; break;
71 case 'g': iMultiplier = 125000000; break;
72 case 'm': iMultiplier = 125000; break;
73 case 'k': iMultiplier = 125; break;
74 default: return "Invalid unit suffix. Valid suffixes are: k, m, g, K, M, G\n";
75 }
76 break;
77 case VWRN_TRAILING_SPACES:
78 return "Trailing spaces in limit!\n";
79 case VERR_NO_DIGITS:
80 return "No digits in limit specifier\n";
81 default:
82 return "Invalid limit specifier\n";
83 }
84 if (*pLimit < 0)
85 return "Limit cannot be negative\n";
86 if (*pLimit > INT64_MAX / iMultiplier)
87 return "Limit is too big\n";
88 *pLimit *= iMultiplier;
89
90 return NULL;
91}
92
93/**
94 * Handles the 'bandwidthctl myvm add' sub-command.
95 * @returns Exit code.
96 * @param a The handler argument package.
97 * @param bwCtrl Reference to the bandwidth control interface.
98 */
99static RTEXITCODE handleBandwidthControlAdd(HandlerArg *a, ComPtr<IBandwidthControl> &bwCtrl)
100{
101 HRESULT rc = S_OK;
102 static const RTGETOPTDEF g_aBWCtlAddOptions[] =
103 {
104 { "--type", 't', RTGETOPT_REQ_STRING },
105 { "--limit", 'l', RTGETOPT_REQ_STRING }
106 };
107
108
109 Bstr name(a->argv[2]);
110 if (name.isEmpty())
111 {
112 errorArgument("Bandwidth group name must not be empty!\n");
113 return RTEXITCODE_FAILURE;
114 }
115
116 const char *pszType = NULL;
117 int64_t cMaxBytesPerSec = INT64_MAX;
118
119 int c;
120 RTGETOPTUNION ValueUnion;
121 RTGETOPTSTATE GetState;
122 RTGetOptInit(&GetState, a->argc, a->argv, g_aBWCtlAddOptions,
123 RT_ELEMENTS(g_aBWCtlAddOptions), 3, RTGETOPTINIT_FLAGS_NO_STD_OPTS);
124
125 while ( SUCCEEDED(rc)
126 && (c = RTGetOpt(&GetState, &ValueUnion)))
127 {
128 switch (c)
129 {
130 case 't': // bandwidth group type
131 {
132 if (ValueUnion.psz)
133 pszType = ValueUnion.psz;
134 else
135 rc = E_FAIL;
136 break;
137 }
138
139 case 'l': // limit
140 {
141 if (ValueUnion.psz)
142 {
143 const char *pcszError = parseLimit(ValueUnion.psz, &cMaxBytesPerSec);
144 if (pcszError)
145 {
146 errorArgument(pcszError);
147 return RTEXITCODE_FAILURE;
148 }
149 }
150 else
151 rc = E_FAIL;
152 break;
153 }
154
155 default:
156 {
157 errorGetOpt(USAGE_BANDWIDTHCONTROL, c, &ValueUnion);
158 rc = E_FAIL;
159 break;
160 }
161 }
162 }
163
164 BandwidthGroupType_T enmType;
165
166 if (!RTStrICmp(pszType, "disk"))
167 enmType = BandwidthGroupType_Disk;
168 else if (!RTStrICmp(pszType, "network"))
169 enmType = BandwidthGroupType_Network;
170 else
171 {
172 errorArgument("Invalid bandwidth group type\n");
173 return RTEXITCODE_FAILURE;
174 }
175
176 CHECK_ERROR2_RET(bwCtrl, CreateBandwidthGroup(name.raw(), enmType, (LONG64)cMaxBytesPerSec), RTEXITCODE_FAILURE);
177
178 return RTEXITCODE_SUCCESS;
179}
180
181/**
182 * Handles the 'bandwidthctl myvm set' sub-command.
183 * @returns Exit code.
184 * @param a The handler argument package.
185 * @param bwCtrl Reference to the bandwidth control interface.
186 */
187static RTEXITCODE handleBandwidthControlSet(HandlerArg *a, ComPtr<IBandwidthControl> &bwCtrl)
188{
189 HRESULT rc = S_OK;
190 static const RTGETOPTDEF g_aBWCtlAddOptions[] =
191 {
192 { "--limit", 'l', RTGETOPT_REQ_STRING }
193 };
194
195
196 Bstr name(a->argv[2]);
197 int64_t cMaxBytesPerSec = INT64_MAX;
198
199 int c;
200 RTGETOPTUNION ValueUnion;
201 RTGETOPTSTATE GetState;
202 RTGetOptInit(&GetState, a->argc, a->argv, g_aBWCtlAddOptions,
203 RT_ELEMENTS(g_aBWCtlAddOptions), 3, RTGETOPTINIT_FLAGS_NO_STD_OPTS);
204
205 while ( SUCCEEDED(rc)
206 && (c = RTGetOpt(&GetState, &ValueUnion)))
207 {
208 switch (c)
209 {
210 case 'l': // limit
211 {
212 if (ValueUnion.psz)
213 {
214 const char *pcszError = parseLimit(ValueUnion.psz, &cMaxBytesPerSec);
215 if (pcszError)
216 {
217 errorArgument(pcszError);
218 return RTEXITCODE_FAILURE;
219 }
220 }
221 else
222 rc = E_FAIL;
223 break;
224 }
225
226 default:
227 {
228 errorGetOpt(USAGE_BANDWIDTHCONTROL, c, &ValueUnion);
229 rc = E_FAIL;
230 break;
231 }
232 }
233 }
234
235
236 if (cMaxBytesPerSec != INT64_MAX)
237 {
238 ComPtr<IBandwidthGroup> bwGroup;
239 CHECK_ERROR2_RET(bwCtrl, GetBandwidthGroup(name.raw(), bwGroup.asOutParam()), RTEXITCODE_FAILURE);
240 if (SUCCEEDED(rc))
241 CHECK_ERROR2_RET(bwGroup, COMSETTER(MaxBytesPerSec)((LONG64)cMaxBytesPerSec), RTEXITCODE_FAILURE);
242 }
243
244 return RTEXITCODE_SUCCESS;
245}
246
247/**
248 * Handles the 'bandwidthctl myvm remove' sub-command.
249 * @returns Exit code.
250 * @param a The handler argument package.
251 * @param bwCtrl Reference to the bandwidth control interface.
252 */
253static RTEXITCODE handleBandwidthControlRemove(HandlerArg *a, ComPtr<IBandwidthControl> &bwCtrl)
254{
255 Bstr name(a->argv[2]);
256 CHECK_ERROR2_RET(bwCtrl, DeleteBandwidthGroup(name.raw()), RTEXITCODE_FAILURE);
257 return RTEXITCODE_SUCCESS;
258}
259
260/**
261 * Handles the 'bandwidthctl myvm list' sub-command.
262 * @returns Exit code.
263 * @param a The handler argument package.
264 * @param bwCtrl Reference to the bandwidth control interface.
265 */
266static RTEXITCODE handleBandwidthControlList(HandlerArg *pArgs, ComPtr<IBandwidthControl> &rptrBWControl)
267{
268 static const RTGETOPTDEF g_aOptions[] =
269 {
270 { "--machinereadable", 'M', RTGETOPT_REQ_NOTHING },
271 };
272
273 VMINFO_DETAILS enmDetails = VMINFO_STANDARD;
274
275 int c;
276 RTGETOPTUNION ValueUnion;
277 RTGETOPTSTATE GetState;
278 RTGetOptInit(&GetState, pArgs->argc, pArgs->argv, g_aOptions, RT_ELEMENTS(g_aOptions), 2 /*iArg*/, 0 /*fFlags*/);
279 while ((c = RTGetOpt(&GetState, &ValueUnion)))
280 {
281 switch (c)
282 {
283 case 'M':
284 enmDetails = VMINFO_MACHINEREADABLE;
285 break;
286 default:
287 return errorGetOpt(USAGE_BANDWIDTHCONTROL, c, &ValueUnion);
288 }
289 }
290
291 /* See showVMInfo. */
292 if (FAILED(showBandwidthGroups(rptrBWControl, enmDetails)))
293 return RTEXITCODE_FAILURE;
294
295 return RTEXITCODE_SUCCESS;
296}
297
298
299/**
300 * Handles the 'bandwidthctl' command.
301 * @returns Exit code.
302 * @param a The handler argument package.
303 */
304int handleBandwidthControl(HandlerArg *a)
305{
306 int c = VERR_INTERNAL_ERROR; /* initialized to shut up gcc */
307 HRESULT rc = S_OK;
308 ComPtr<IMachine> machine;
309 ComPtr<IBandwidthControl> bwCtrl;
310
311 if (a->argc < 2)
312 return errorSyntax(USAGE_BANDWIDTHCONTROL, "Too few parameters");
313 else if (a->argc > 7)
314 return errorSyntax(USAGE_BANDWIDTHCONTROL, "Too many parameters");
315
316 /* try to find the given machine */
317 CHECK_ERROR_RET(a->virtualBox, FindMachine(Bstr(a->argv[0]).raw(),
318 machine.asOutParam()), 1);
319
320 /* open a session for the VM (new or shared) */
321 CHECK_ERROR_RET(machine, LockMachine(a->session, LockType_Shared), 1);
322 SessionType_T st;
323 CHECK_ERROR_RET(a->session, COMGETTER(Type)(&st), 1);
324 bool fRunTime = (st == SessionType_Shared);
325
326 /* get the mutable session machine */
327 a->session->COMGETTER(Machine)(machine.asOutParam());
328 rc = machine->COMGETTER(BandwidthControl)(bwCtrl.asOutParam());
329 if (FAILED(rc)) goto leave;
330
331 if (!strcmp(a->argv[1], "add"))
332 {
333 if (fRunTime)
334 {
335 errorArgument("Bandwidth groups cannot be created while the VM is running\n");
336 goto leave;
337 }
338 rc = handleBandwidthControlAdd(a, bwCtrl) == RTEXITCODE_SUCCESS ? S_OK : E_FAIL;
339 }
340 else if (!strcmp(a->argv[1], "remove"))
341 {
342 if (fRunTime)
343 {
344 errorArgument("Bandwidth groups cannot be deleted while the VM is running\n");
345 goto leave;
346 }
347 rc = handleBandwidthControlRemove(a, bwCtrl) == RTEXITCODE_SUCCESS ? S_OK : E_FAIL;
348 }
349 else if (!strcmp(a->argv[1], "set"))
350 rc = handleBandwidthControlSet(a, bwCtrl) == RTEXITCODE_SUCCESS ? S_OK : E_FAIL;
351 else if (!strcmp(a->argv[1], "list"))
352 rc = handleBandwidthControlList(a, bwCtrl) == RTEXITCODE_SUCCESS ? S_OK : E_FAIL;
353 else
354 {
355 errorSyntax(USAGE_BANDWIDTHCONTROL, "Invalid parameter '%s'", Utf8Str(a->argv[1]).c_str());
356 rc = E_FAIL;
357 }
358
359 /* commit changes */
360 if (SUCCEEDED(rc))
361 CHECK_ERROR(machine, SaveSettings());
362
363leave:
364 /* it's important to always close sessions */
365 a->session->UnlockMachine();
366
367 return SUCCEEDED(rc) ? 0 : 1;
368}
369
370#endif /* !VBOX_ONLY_DOCS */
371
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