VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/NetworkAdapterImpl.cpp@ 61173

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

Main/NetworkAdapter: one more RT_BOOL() to fix vcc warning

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 39.4 KB
Line 
1/* $Id: NetworkAdapterImpl.cpp 61173 2016-05-24 17:23:58Z vboxsync $ */
2/** @file
3 * Implementation of INetworkAdapter in VBoxSVC.
4 */
5
6/*
7 * Copyright (C) 2006-2016 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 "NetworkAdapterImpl.h"
19#include "NATEngineImpl.h"
20#include "AutoCaller.h"
21#include "Logging.h"
22#include "MachineImpl.h"
23#include "GuestOSTypeImpl.h"
24#include "HostImpl.h"
25#include "SystemPropertiesImpl.h"
26#include "VirtualBoxImpl.h"
27
28#include <iprt/string.h>
29#include <iprt/cpp/utils.h>
30
31#include <VBox/err.h>
32#include <VBox/settings.h>
33
34#include "AutoStateDep.h"
35
36// constructor / destructor
37////////////////////////////////////////////////////////////////////////////////
38
39NetworkAdapter::NetworkAdapter()
40 : mParent(NULL)
41{
42}
43
44NetworkAdapter::~NetworkAdapter()
45{
46}
47
48HRESULT NetworkAdapter::FinalConstruct()
49{
50 return BaseFinalConstruct();
51}
52
53void NetworkAdapter::FinalRelease()
54{
55 uninit();
56 BaseFinalRelease();
57}
58
59// public initializer/uninitializer for internal purposes only
60////////////////////////////////////////////////////////////////////////////////
61
62/**
63 * Initializes the network adapter object.
64 *
65 * @param aParent Handle of the parent object.
66 */
67HRESULT NetworkAdapter::init(Machine *aParent, ULONG uSlot)
68{
69 LogFlowThisFunc(("aParent=%p, uSlot=%d\n", aParent, uSlot));
70
71 ComAssertRet(aParent, E_INVALIDARG);
72 uint32_t maxNetworkAdapters = Global::getMaxNetworkAdapters(aParent->i_getChipsetType());
73 ComAssertRet(uSlot < maxNetworkAdapters, E_INVALIDARG);
74
75 /* Enclose the state transition NotReady->InInit->Ready */
76 AutoInitSpan autoInitSpan(this);
77 AssertReturn(autoInitSpan.isOk(), E_FAIL);
78
79 unconst(mParent) = aParent;
80 unconst(mNATEngine).createObject();
81 mNATEngine->init(aParent, this);
82 /* mPeer is left null */
83
84 mData.allocate();
85
86 /* initialize data */
87 mData->ulSlot = uSlot;
88
89 /* default to Am79C973 */
90 mData->type = NetworkAdapterType_Am79C973;
91
92 /* Confirm a successful initialization */
93 autoInitSpan.setSucceeded();
94
95 return S_OK;
96}
97
98/**
99 * Initializes the network adapter object given another network adapter object
100 * (a kind of copy constructor). This object shares data with
101 * the object passed as an argument.
102 *
103 * @param aReshare
104 * When false, the original object will remain a data owner.
105 * Otherwise, data ownership will be transferred from the original
106 * object to this one.
107 *
108 * @note This object must be destroyed before the original object
109 * it shares data with is destroyed.
110 *
111 * @note Locks @a aThat object for reading.
112 */
113HRESULT NetworkAdapter::init(Machine *aParent, NetworkAdapter *aThat, bool aReshare /* = false */)
114{
115 LogFlowThisFunc(("aParent=%p, aThat=%p, aReshare=%RTbool\n", aParent, aThat, aReshare));
116
117 ComAssertRet(aParent && aThat, E_INVALIDARG);
118
119 /* Enclose the state transition NotReady->InInit->Ready */
120 AutoInitSpan autoInitSpan(this);
121 AssertReturn(autoInitSpan.isOk(), E_FAIL);
122
123 unconst(mParent) = aParent;
124 /* mPeer is left null */
125
126 unconst(mNATEngine).createObject();
127 mNATEngine->init(aParent, this, aThat->mNATEngine);
128
129 /* sanity */
130 AutoCaller thatCaller(aThat);
131 AssertComRCReturnRC(thatCaller.rc());
132
133 if (aReshare)
134 {
135 AutoWriteLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
136
137 unconst(aThat->mPeer) = this;
138 mData.attach(aThat->mData);
139 }
140 else
141 {
142 unconst(mPeer) = aThat;
143
144 AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
145 mData.share(aThat->mData);
146 }
147
148 /* Confirm a successful initialization */
149 autoInitSpan.setSucceeded();
150
151 return S_OK;
152}
153
154/**
155 * Initializes the guest object given another guest object
156 * (a kind of copy constructor). This object makes a private copy of data
157 * of the original object passed as an argument.
158 *
159 * @note Locks @a aThat object for reading.
160 */
161HRESULT NetworkAdapter::initCopy(Machine *aParent, NetworkAdapter *aThat)
162{
163 LogFlowThisFunc(("aParent=%p, aThat=%p\n", aParent, aThat));
164
165 ComAssertRet(aParent && aThat, E_INVALIDARG);
166
167 /* Enclose the state transition NotReady->InInit->Ready */
168 AutoInitSpan autoInitSpan(this);
169 AssertReturn(autoInitSpan.isOk(), E_FAIL);
170
171 unconst(mParent) = aParent;
172 /* mPeer is left null */
173
174 unconst(mNATEngine).createObject();
175 mNATEngine->initCopy(aParent, this, aThat->mNATEngine);
176
177 /* sanity */
178 AutoCaller thatCaller(aThat);
179 AssertComRCReturnRC(thatCaller.rc());
180
181 AutoReadLock thatLock(aThat COMMA_LOCKVAL_SRC_POS);
182 mData.attachCopy(aThat->mData);
183
184 /* Confirm a successful initialization */
185 autoInitSpan.setSucceeded();
186
187 return S_OK;
188}
189
190/**
191 * Uninitializes the instance and sets the ready flag to FALSE.
192 * Called either from FinalRelease() or by the parent when it gets destroyed.
193 */
194void NetworkAdapter::uninit()
195{
196 LogFlowThisFunc(("\n"));
197
198 /* Enclose the state transition Ready->InUninit->NotReady */
199 AutoUninitSpan autoUninitSpan(this);
200 if (autoUninitSpan.uninitDone())
201 return;
202
203 mData.free();
204
205 unconst(mNATEngine).setNull();
206 unconst(mPeer) = NULL;
207 unconst(mParent) = NULL;
208}
209
210// wrapped INetworkAdapter properties
211////////////////////////////////////////////////////////////////////////////////
212HRESULT NetworkAdapter::getAdapterType(NetworkAdapterType_T *aAdapterType)
213{
214 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
215
216 *aAdapterType = mData->type;
217
218 return S_OK;
219}
220
221HRESULT NetworkAdapter::setAdapterType(NetworkAdapterType_T aAdapterType)
222{
223 /* the machine needs to be mutable */
224 AutoMutableStateDependency adep(mParent);
225 if (FAILED(adep.rc())) return adep.rc();
226
227 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
228
229 /* make sure the value is allowed */
230 switch (aAdapterType)
231 {
232 case NetworkAdapterType_Am79C970A:
233 case NetworkAdapterType_Am79C973:
234#ifdef VBOX_WITH_E1000
235 case NetworkAdapterType_I82540EM:
236 case NetworkAdapterType_I82543GC:
237 case NetworkAdapterType_I82545EM:
238#endif
239#ifdef VBOX_WITH_VIRTIO
240 case NetworkAdapterType_Virtio:
241#endif /* VBOX_WITH_VIRTIO */
242 break;
243 default:
244 return setError(E_FAIL,
245 tr("Invalid network adapter type '%d'"),
246 aAdapterType);
247 }
248
249 if (mData->type != aAdapterType)
250 {
251 mData.backup();
252 mData->type = aAdapterType;
253
254 // leave the lock before informing callbacks
255 alock.release();
256
257 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
258 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
259 mlock.release();
260
261 /* Changing the network adapter type during runtime is not allowed,
262 * therefore no immediate change in CFGM logic => changeAdapter=FALSE. */
263 mParent->i_onNetworkAdapterChange(this, FALSE);
264 }
265
266 return S_OK;
267}
268
269
270HRESULT NetworkAdapter::getSlot(ULONG *uSlot)
271{
272 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
273
274 *uSlot = mData->ulSlot;
275
276 return S_OK;
277}
278
279HRESULT NetworkAdapter::getEnabled(BOOL *aEnabled)
280{
281 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
282
283 *aEnabled = mData->fEnabled;
284
285 return S_OK;
286}
287
288HRESULT NetworkAdapter::setEnabled(BOOL aEnabled)
289{
290 /* the machine needs to be mutable */
291 AutoMutableStateDependency adep(mParent);
292 if (FAILED(adep.rc())) return adep.rc();
293
294 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
295
296 if (mData->fEnabled != RT_BOOL(aEnabled))
297 {
298 mData.backup();
299 mData->fEnabled = RT_BOOL(aEnabled);
300 if (RT_BOOL(aEnabled) && mData->strMACAddress.isEmpty())
301 i_generateMACAddress();
302
303 // leave the lock before informing callbacks
304 alock.release();
305
306 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
307 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
308 mlock.release();
309
310 /* Disabling the network adapter during runtime is not allowed
311 * therefore no immediate change in CFGM logic => changeAdapter=FALSE. */
312 mParent->i_onNetworkAdapterChange(this, FALSE);
313 }
314
315 return S_OK;
316}
317
318HRESULT NetworkAdapter::getMACAddress(com::Utf8Str &aMACAddress)
319{
320 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
321
322 ComAssertRet(!mData->fEnabled || !mData->strMACAddress.isEmpty(), E_FAIL);
323
324 aMACAddress = mData->strMACAddress;
325
326 return S_OK;
327}
328
329HRESULT NetworkAdapter::i_updateMacAddress(Utf8Str aMACAddress)
330{
331 HRESULT rc = S_OK;
332
333 /*
334 * Are we supposed to generate a MAC?
335 */
336 if (mData->fEnabled && aMACAddress.isEmpty())
337 i_generateMACAddress();
338 else
339 {
340 if (mData->strMACAddress != aMACAddress)
341 {
342 if (mData->fEnabled || !aMACAddress.isEmpty())
343 {
344 /*
345 * Verify given MAC address
346 */
347 char *macAddressStr = aMACAddress.mutableRaw();
348 int i = 0;
349 while ((i < 13) && macAddressStr && *macAddressStr && (rc == S_OK))
350 {
351 char c = *macAddressStr;
352 /* canonicalize hex digits to capital letters */
353 if (c >= 'a' && c <= 'f')
354 {
355 /** @todo the runtime lacks an ascii lower/upper conv */
356 c &= 0xdf;
357 *macAddressStr = c;
358 }
359 /* we only accept capital letters */
360 if (((c < '0') || (c > '9')) &&
361 ((c < 'A') || (c > 'F')))
362 rc = setError(E_INVALIDARG, tr("Invalid MAC address format"));
363 /* the second digit must have even value for unicast addresses */
364 if ((i == 1) && (!!(c & 1) == (c >= '0' && c <= '9')))
365 rc = setError(E_INVALIDARG, tr("Invalid MAC address format"));
366
367 macAddressStr++;
368 i++;
369 }
370 /* we must have parsed exactly 12 characters */
371 if (i != 12)
372 rc = setError(E_INVALIDARG, tr("Invalid MAC address format"));
373 }
374
375 if (SUCCEEDED(rc))
376 mData->strMACAddress = aMACAddress;
377 }
378 }
379
380 return rc;
381}
382
383HRESULT NetworkAdapter::setMACAddress(const com::Utf8Str &aMACAddress)
384{
385 /* the machine needs to be mutable */
386 AutoMutableStateDependency adep(mParent);
387 if (FAILED(adep.rc())) return adep.rc();
388
389 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
390 mData.backup();
391
392 HRESULT rc = i_updateMacAddress(aMACAddress);
393 if (SUCCEEDED(rc))
394 {
395 // leave the lock before informing callbacks
396 alock.release();
397
398 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
399 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
400 mlock.release();
401
402 /* Changing the MAC via the Main API during runtime is not allowed,
403 * therefore no immediate change in CFGM logic => changeAdapter=FALSE. */
404 mParent->i_onNetworkAdapterChange(this, FALSE);
405 }
406
407 return rc;
408}
409
410HRESULT NetworkAdapter::getAttachmentType(NetworkAttachmentType_T *aAttachmentType)
411{
412 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
413
414 *aAttachmentType = mData->mode;
415
416 return S_OK;
417}
418
419HRESULT NetworkAdapter::setAttachmentType(NetworkAttachmentType_T aAttachmentType)
420{
421 /* the machine needs to be mutable */
422 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
423 if (FAILED(adep.rc())) return adep.rc();
424
425 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
426
427 if (mData->mode != aAttachmentType)
428 {
429 mData.backup();
430
431 /* there must an internal network name */
432 if (mData->strInternalNetworkName.isEmpty())
433 {
434 Log(("Internal network name not defined, setting to default \"intnet\"\n"));
435 mData->strInternalNetworkName = "intnet";
436 }
437
438 /* there must a NAT network name */
439 if (mData->strNATNetworkName.isEmpty())
440 {
441 Log(("NAT network name not defined, setting to default \"NatNetwork\"\n"));
442 mData->strNATNetworkName = "NatNetwork";
443 }
444
445 NetworkAttachmentType_T oldAttachmentType = mData->mode;
446 mData->mode = aAttachmentType;
447
448 // leave the lock before informing callbacks
449 alock.release();
450
451 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
452 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
453 mlock.release();
454
455 if (oldAttachmentType == NetworkAttachmentType_NATNetwork)
456 i_checkAndSwitchFromNatNetworking(mData->strNATNetworkName);
457
458 if (aAttachmentType == NetworkAttachmentType_NATNetwork)
459 i_switchToNatNetworking(mData->strNATNetworkName);
460
461 /* Adapt the CFGM logic and notify the guest => changeAdapter=TRUE. */
462 mParent->i_onNetworkAdapterChange(this, TRUE);
463 }
464
465 return S_OK;
466}
467
468HRESULT NetworkAdapter::getBridgedInterface(com::Utf8Str &aBridgedInterface)
469{
470 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
471
472 aBridgedInterface = mData->strBridgedName;
473
474 return S_OK;
475}
476
477HRESULT NetworkAdapter::setBridgedInterface(const com::Utf8Str &aBridgedInterface)
478{
479 /* the machine needs to be mutable */
480 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
481 if (FAILED(adep.rc())) return adep.rc();
482
483 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
484
485 if (mData->strBridgedName != aBridgedInterface)
486 {
487 /* if an empty/null string is to be set, bridged interface must be
488 * turned off */
489 if (aBridgedInterface.isEmpty()
490 && mData->mode == NetworkAttachmentType_Bridged)
491 {
492 return setError(E_FAIL,
493 tr("Empty or null bridged interface name is not valid"));
494 }
495
496 mData.backup();
497 mData->strBridgedName = aBridgedInterface;
498
499 // leave the lock before informing callbacks
500 alock.release();
501
502 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
503 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
504 mlock.release();
505
506 /* When changing the host adapter, adapt the CFGM logic to make this
507 * change immediately effect and to notify the guest that the network
508 * might have changed, therefore changeAdapter=TRUE. */
509 mParent->i_onNetworkAdapterChange(this, TRUE);
510 }
511
512 return S_OK;
513}
514
515HRESULT NetworkAdapter::getHostOnlyInterface(com::Utf8Str &aHostOnlyInterface)
516{
517 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
518
519 aHostOnlyInterface = mData->strHostOnlyName;
520
521 return S_OK;
522}
523
524HRESULT NetworkAdapter::setHostOnlyInterface(const com::Utf8Str &aHostOnlyInterface)
525{
526 /* the machine needs to be mutable */
527 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
528 if (FAILED(adep.rc())) return adep.rc();
529
530 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
531
532 if (mData->strHostOnlyName != aHostOnlyInterface)
533 {
534 /* if an empty/null string is to be set, host only interface must be
535 * turned off */
536 if ( aHostOnlyInterface.isEmpty()
537 && mData->mode == NetworkAttachmentType_HostOnly)
538 {
539 return setError(E_FAIL,
540 tr("Empty or null host only interface name is not valid"));
541 }
542
543 mData.backup();
544 mData->strHostOnlyName = aHostOnlyInterface;
545
546 // leave the lock before informing callbacks
547 alock.release();
548
549 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
550 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
551 mlock.release();
552
553 /* When changing the host adapter, adapt the CFGM logic to make this
554 * change immediately effect and to notify the guest that the network
555 * might have changed, therefore changeAdapter=TRUE. */
556 mParent->i_onNetworkAdapterChange(this, TRUE);
557 }
558
559 return S_OK;
560}
561
562
563HRESULT NetworkAdapter::getInternalNetwork(com::Utf8Str &aInternalNetwork)
564{
565 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
566
567 aInternalNetwork = mData->strInternalNetworkName;
568
569 return S_OK;
570}
571
572HRESULT NetworkAdapter::setInternalNetwork(const com::Utf8Str &aInternalNetwork)
573{
574 /* the machine needs to be mutable */
575 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
576 if (FAILED(adep.rc())) return adep.rc();
577
578 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
579
580 if (mData->strInternalNetworkName != aInternalNetwork)
581 {
582 /* if an empty/null string is to be set, internal networking must be
583 * turned off */
584 if (aInternalNetwork.isEmpty() && mData->mode == NetworkAttachmentType_Internal)
585 {
586 return setError(E_FAIL,
587 tr("Empty or null internal network name is not valid"));
588 }
589 mData.backup();
590 mData->strInternalNetworkName = aInternalNetwork;
591
592 // leave the lock before informing callbacks
593 alock.release();
594
595 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
596 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
597 mlock.release();
598
599 /* When changing the internal network, adapt the CFGM logic to make this
600 * change immediately effect and to notify the guest that the network
601 * might have changed, therefore changeAdapter=TRUE. */
602 mParent->i_onNetworkAdapterChange(this, TRUE);
603 }
604
605 return S_OK;
606}
607
608HRESULT NetworkAdapter::getNATNetwork(com::Utf8Str &aNATNetwork)
609{
610 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
611
612 aNATNetwork = mData->strNATNetworkName;
613
614 return S_OK;
615}
616
617
618HRESULT NetworkAdapter::setNATNetwork(const com::Utf8Str &aNATNetwork)
619{
620 /* the machine needs to be mutable */
621 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
622 if (FAILED(adep.rc())) return adep.rc();
623
624 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
625
626 if (mData->strNATNetworkName != aNATNetwork)
627 {
628 /* if an empty/null string is to be set, host only interface must be
629 * turned off */
630 if (aNATNetwork.isEmpty()
631 && mData->mode == NetworkAttachmentType_NATNetwork)
632 return setError(E_FAIL,
633 tr("Empty or null NAT network name is not valid"));
634
635 mData.backup();
636
637 Bstr oldNatNetworkName = mData->strNATNetworkName;
638 mData->strNATNetworkName = aNATNetwork;
639
640 // leave the lock before informing callbacks
641 alock.release();
642
643 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
644 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
645 mlock.release();
646 i_checkAndSwitchFromNatNetworking(oldNatNetworkName.raw());
647
648 i_switchToNatNetworking(aNATNetwork);
649 /* When changing the host adapter, adapt the CFGM logic to make this
650 * change immediately effect and to notify the guest that the network
651 * might have changed, therefore changeAdapter=TRUE. */
652 mParent->i_onNetworkAdapterChange(this, TRUE);
653 }
654
655 return S_OK;
656}
657
658HRESULT NetworkAdapter::getGenericDriver(com::Utf8Str &aGenericDriver)
659{
660 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
661
662 aGenericDriver = mData->strGenericDriver;
663
664 return S_OK;
665}
666
667HRESULT NetworkAdapter::setGenericDriver(const com::Utf8Str &aGenericDriver)
668{
669 /* the machine needs to be mutable */
670 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
671 if (FAILED(adep.rc())) return adep.rc();
672
673 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
674
675 if (mData->strGenericDriver != aGenericDriver)
676 {
677 mData.backup();
678 mData->strGenericDriver = aGenericDriver;
679
680 /* leave the lock before informing callbacks */
681 alock.release();
682
683 mParent->i_onNetworkAdapterChange(this, FALSE);
684 }
685
686 return S_OK;
687}
688
689
690HRESULT NetworkAdapter::getCableConnected(BOOL *aConnected)
691{
692 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
693
694 *aConnected = mData->fCableConnected;
695
696 return S_OK;
697}
698
699
700HRESULT NetworkAdapter::setCableConnected(BOOL aConnected)
701{
702 /* the machine needs to be mutable */
703 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
704 if (FAILED(adep.rc())) return adep.rc();
705
706 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
707
708 if (RT_BOOL(aConnected) != mData->fCableConnected)
709 {
710 mData.backup();
711 mData->fCableConnected = RT_BOOL(aConnected);
712
713 // leave the lock before informing callbacks
714 alock.release();
715
716 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
717 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
718 mlock.release();
719
720 /* No change in CFGM logic => changeAdapter=FALSE. */
721 mParent->i_onNetworkAdapterChange(this, FALSE);
722 }
723
724 return S_OK;
725}
726
727
728HRESULT NetworkAdapter::getLineSpeed(ULONG *aSpeed)
729{
730 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
731
732 *aSpeed = mData->ulLineSpeed;
733
734 return S_OK;
735}
736
737HRESULT NetworkAdapter::setLineSpeed(ULONG aSpeed)
738{
739 /* the machine needs to be mutable */
740 AutoMutableStateDependency adep(mParent);
741 if (FAILED(adep.rc())) return adep.rc();
742
743 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
744
745 if (aSpeed != mData->ulLineSpeed)
746 {
747 mData.backup();
748 mData->ulLineSpeed = aSpeed;
749
750 // leave the lock before informing callbacks
751 alock.release();
752
753 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
754 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
755 mlock.release();
756
757 /* No change in CFGM logic => changeAdapter=FALSE. */
758 mParent->i_onNetworkAdapterChange(this, FALSE);
759 }
760
761 return S_OK;
762}
763
764HRESULT NetworkAdapter::getPromiscModePolicy(NetworkAdapterPromiscModePolicy_T *aPromiscModePolicy)
765{
766 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
767
768 *aPromiscModePolicy = mData->enmPromiscModePolicy;
769
770 return S_OK;
771}
772
773HRESULT NetworkAdapter::setPromiscModePolicy(NetworkAdapterPromiscModePolicy_T aPromiscModePolicy)
774{
775 /* the machine needs to be mutable */
776 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
777 if (FAILED(adep.rc())) return adep.rc();
778
779 switch (aPromiscModePolicy)
780 {
781 case NetworkAdapterPromiscModePolicy_Deny:
782 case NetworkAdapterPromiscModePolicy_AllowNetwork:
783 case NetworkAdapterPromiscModePolicy_AllowAll:
784 break;
785 default:
786 return setError(E_INVALIDARG, tr("Invalid promiscuous mode policy (%d)"), aPromiscModePolicy);
787 }
788
789 AutoCaller autoCaller(this);
790 HRESULT hrc = autoCaller.rc();
791
792 if (SUCCEEDED(hrc))
793 {
794 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
795 if (aPromiscModePolicy != mData->enmPromiscModePolicy)
796 {
797 mData.backup();
798 mData->enmPromiscModePolicy = aPromiscModePolicy;
799
800 alock.release();
801 mParent->i_setModifiedLock(Machine::IsModified_NetworkAdapters);
802 mParent->i_onNetworkAdapterChange(this, TRUE);
803 }
804 }
805
806 return hrc;
807}
808
809
810HRESULT NetworkAdapter::getTraceEnabled(BOOL *aEnabled)
811{
812
813 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
814
815 *aEnabled = mData->fTraceEnabled;
816
817 return S_OK;
818}
819
820HRESULT NetworkAdapter::setTraceEnabled(BOOL aEnabled)
821{
822 /* the machine needs to be mutable */
823 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
824 if (FAILED(adep.rc())) return adep.rc();
825
826 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
827
828 if (RT_BOOL(aEnabled) != mData->fTraceEnabled)
829 {
830 mData.backup();
831 mData->fTraceEnabled = RT_BOOL(aEnabled);
832
833 // leave the lock before informing callbacks
834 alock.release();
835
836 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
837 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
838 mlock.release();
839
840 /* Adapt the CFGM logic changeAdapter=TRUE */
841 mParent->i_onNetworkAdapterChange(this, TRUE);
842 }
843
844 return S_OK;
845}
846
847HRESULT NetworkAdapter::getTraceFile(com::Utf8Str &aTraceFile)
848{
849 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
850
851 aTraceFile = mData->strTraceFile;
852
853 return S_OK;
854}
855
856
857HRESULT NetworkAdapter::setTraceFile(const com::Utf8Str &aTraceFile)
858{
859 /* the machine needs to be mutable */
860 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
861 if (FAILED(adep.rc())) return adep.rc();
862
863 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
864
865 if (mData->strTraceFile != aTraceFile)
866 {
867 mData.backup();
868 mData->strTraceFile = aTraceFile;
869
870 // leave the lock before informing callbacks
871 alock.release();
872
873 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
874 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
875 mlock.release();
876
877 /* We change the 'File' => changeAdapter=TRUE. */
878 mParent->i_onNetworkAdapterChange(this, TRUE);
879 }
880
881 return S_OK;
882}
883
884HRESULT NetworkAdapter::getNATEngine(ComPtr<INATEngine> &aNATEngine)
885{
886 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
887
888 aNATEngine = mNATEngine;
889
890 return S_OK;
891}
892
893HRESULT NetworkAdapter::getBootPriority(ULONG *aBootPriority)
894{
895 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
896
897 *aBootPriority = mData->ulBootPriority;
898
899 return S_OK;
900}
901
902HRESULT NetworkAdapter::setBootPriority(ULONG aBootPriority)
903{
904 /* the machine needs to be mutable */
905 AutoMutableStateDependency adep(mParent);
906 if (FAILED(adep.rc())) return adep.rc();
907
908 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
909
910 if (aBootPriority != mData->ulBootPriority)
911 {
912 mData.backup();
913 mData->ulBootPriority = aBootPriority;
914
915 // leave the lock before informing callbacks
916 alock.release();
917
918 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS); // mParent is const, no need to lock
919 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
920 mlock.release();
921
922 /* No change in CFGM logic => changeAdapter=FALSE. */
923 mParent->i_onNetworkAdapterChange(this, FALSE);
924 }
925
926 return S_OK;
927}
928
929// wrapped INetworkAdapter methods
930////////////////////////////////////////////////////////////////////////////////
931
932HRESULT NetworkAdapter::getProperty(const com::Utf8Str &aKey, com::Utf8Str &aValue)
933{
934 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
935 aValue = "";
936 settings::StringsMap::const_iterator it = mData->genericProperties.find(aKey);
937 if (it != mData->genericProperties.end())
938 aValue = it->second; // source is a Utf8Str
939
940 return S_OK;
941}
942
943HRESULT NetworkAdapter::setProperty(const com::Utf8Str &aKey, const com::Utf8Str &aValue)
944{
945 LogFlowThisFunc(("\n"));
946 /* The machine needs to be mutable. */
947 AutoMutableOrSavedOrRunningStateDependency adep(mParent);
948 if (FAILED(adep.rc())) return adep.rc();
949 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
950 bool fGenericChange = (mData->mode == NetworkAttachmentType_Generic);
951 /* Generic properties processing.
952 * Look up the old value first; if nothing's changed then do nothing.
953 */
954 Utf8Str strOldValue;
955 settings::StringsMap::const_iterator it = mData->genericProperties.find(aKey);
956 if (it != mData->genericProperties.end())
957 strOldValue = it->second;
958
959 if (strOldValue != aValue)
960 {
961 if (aValue.isEmpty())
962 mData->genericProperties.erase(aKey);
963 else
964 mData->genericProperties[aKey] = aValue;
965
966 /* leave the lock before informing callbacks */
967 alock.release();
968
969 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS);
970 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
971 mlock.release();
972
973 /* Avoid deadlock when the event triggers a call to a method of this
974 * interface. */
975 adep.release();
976
977 mParent->i_onNetworkAdapterChange(this, fGenericChange);
978 }
979
980 return S_OK;
981}
982
983HRESULT NetworkAdapter::getProperties(const com::Utf8Str &aNames,
984 std::vector<com::Utf8Str> &aReturnNames,
985 std::vector<com::Utf8Str> &aReturnValues)
986{
987 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
988
989 /// @todo make use of aNames according to the documentation
990 NOREF(aNames);
991 aReturnNames.resize(mData->genericProperties.size());
992 aReturnValues.resize(mData->genericProperties.size());
993
994 size_t i = 0;
995
996 for (settings::StringsMap::const_iterator it = mData->genericProperties.begin();
997 it != mData->genericProperties.end();
998 ++it, ++i)
999 {
1000 aReturnNames[i] = it->first;
1001 aReturnValues[i] = it->second;
1002 }
1003
1004 return S_OK;
1005}
1006
1007
1008
1009// public methods only for internal purposes
1010////////////////////////////////////////////////////////////////////////////////
1011
1012/**
1013 * Loads settings from the given adapter node.
1014 * May be called once right after this object creation.
1015 *
1016 * @param aAdapterNode <Adapter> node.
1017 *
1018 * @note Locks this object for writing.
1019 */
1020HRESULT NetworkAdapter::i_loadSettings(BandwidthControl *bwctl,
1021 const settings::NetworkAdapter &data)
1022{
1023 AutoCaller autoCaller(this);
1024 AssertComRCReturnRC(autoCaller.rc());
1025
1026 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1027
1028 /* Note: we assume that the default values for attributes of optional
1029 * nodes are assigned in the Data::Data() constructor and don't do it
1030 * here. It implies that this method may only be called after constructing
1031 * a new BIOSSettings object while all its data fields are in the default
1032 * values. Exceptions are fields whose creation time defaults don't match
1033 * values that should be applied when these fields are not explicitly set
1034 * in the settings file (for backwards compatibility reasons). This takes
1035 * place when a setting of a newly created object must default to A while
1036 * the same setting of an object loaded from the old settings file must
1037 * default to B. */
1038
1039 HRESULT rc = S_OK;
1040
1041 /* MAC address (can be null) */
1042 rc = i_updateMacAddress(data.strMACAddress);
1043 if (FAILED(rc)) return rc;
1044
1045 mData.assignCopy(&data);
1046
1047 if (mData->strBandwidthGroup.isNotEmpty())
1048 {
1049 ComObjPtr<BandwidthGroup> group;
1050 rc = bwctl->i_getBandwidthGroupByName(data.strBandwidthGroup, group, true);
1051 if (FAILED(rc)) return rc;
1052 group->i_reference();
1053 }
1054
1055 // Load NAT engine settings.
1056 mNATEngine->i_loadSettings(data.nat);
1057
1058 // leave the lock before setting attachment type
1059 alock.release();
1060
1061 rc = COMSETTER(AttachmentType)(data.mode);
1062 if (FAILED(rc)) return rc;
1063
1064 return S_OK;
1065}
1066
1067/**
1068 * Saves settings to the given adapter node.
1069 *
1070 * Note that the given Adapter node is completely empty on input.
1071 *
1072 * @param aAdapterNode <Adapter> node.
1073 *
1074 * @note Locks this object for reading.
1075 */
1076HRESULT NetworkAdapter::i_saveSettings(settings::NetworkAdapter &data)
1077{
1078 AutoCaller autoCaller(this);
1079 AssertComRCReturnRC(autoCaller.rc());
1080
1081 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1082
1083 data = *mData.data();
1084
1085 mNATEngine->i_saveSettings(data.nat);
1086
1087 return S_OK;
1088}
1089
1090/**
1091 * Returns true if any setter method has modified settings of this instance.
1092 * @return
1093 */
1094bool NetworkAdapter::i_isModified() {
1095
1096 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1097
1098 bool fChanged = mData.isBackedUp();
1099 fChanged |= (mData->type == NetworkAttachmentType_NAT? mNATEngine->i_isModified() : false);
1100 return fChanged;
1101}
1102
1103/**
1104 * @note Locks this object for writing.
1105 */
1106void NetworkAdapter::i_rollback()
1107{
1108 /* sanity */
1109 AutoCaller autoCaller(this);
1110 AssertComRCReturnVoid(autoCaller.rc());
1111
1112 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1113
1114 mNATEngine->i_rollback();
1115
1116 mData.rollback();
1117}
1118
1119/**
1120 * @note Locks this object for writing, together with the peer object (also
1121 * for writing) if there is one.
1122 */
1123void NetworkAdapter::i_commit()
1124{
1125 /* sanity */
1126 AutoCaller autoCaller(this);
1127 AssertComRCReturnVoid(autoCaller.rc());
1128
1129 /* sanity too */
1130 AutoCaller peerCaller(mPeer);
1131 AssertComRCReturnVoid(peerCaller.rc());
1132
1133 /* lock both for writing since we modify both (mPeer is "master" so locked
1134 * first) */
1135 AutoMultiWriteLock2 alock(mPeer, this COMMA_LOCKVAL_SRC_POS);
1136
1137 mNATEngine->i_commit();
1138
1139 if (mData.isBackedUp())
1140 {
1141 mData.commit();
1142 if (mPeer)
1143 {
1144 /* attach new data to the peer and reshare it */
1145 mPeer->mData.attach(mData);
1146 }
1147 }
1148}
1149
1150/**
1151 * @note Locks this object for writing, together with the peer object
1152 * represented by @a aThat (locked for reading).
1153 */
1154void NetworkAdapter::i_copyFrom(NetworkAdapter *aThat)
1155{
1156 AssertReturnVoid(aThat != NULL);
1157
1158 /* sanity */
1159 AutoCaller autoCaller(this);
1160 AssertComRCReturnVoid(autoCaller.rc());
1161
1162 /* sanity too */
1163 AutoCaller thatCaller(aThat);
1164 AssertComRCReturnVoid(thatCaller.rc());
1165
1166 mNATEngine->i_copyFrom(aThat->mNATEngine);
1167
1168 /* peer is not modified, lock it for reading (aThat is "master" so locked
1169 * first) */
1170 AutoReadLock rl(aThat COMMA_LOCKVAL_SRC_POS);
1171 AutoWriteLock wl(this COMMA_LOCKVAL_SRC_POS);
1172
1173 /* this will back up current data */
1174 mData.assignCopy(aThat->mData);
1175
1176}
1177
1178void NetworkAdapter::i_applyDefaults(GuestOSType *aOsType)
1179{
1180 AssertReturnVoid(aOsType != NULL);
1181
1182 /* sanity */
1183 AutoCaller autoCaller(this);
1184 AssertComRCReturnVoid(autoCaller.rc());
1185
1186 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1187
1188 bool e1000enabled = false;
1189#ifdef VBOX_WITH_E1000
1190 e1000enabled = true;
1191#endif // VBOX_WITH_E1000
1192
1193 NetworkAdapterType_T defaultType = aOsType->i_networkAdapterType();
1194
1195 /* Set default network adapter for this OS type */
1196 if (defaultType == NetworkAdapterType_I82540EM ||
1197 defaultType == NetworkAdapterType_I82543GC ||
1198 defaultType == NetworkAdapterType_I82545EM)
1199 {
1200 if (e1000enabled) mData->type = defaultType;
1201 }
1202 else mData->type = defaultType;
1203
1204 /* Enable the first one adapter to the NAT */
1205 if (mData->ulSlot == 0)
1206 {
1207 mData->fEnabled = true;
1208 if (mData->strMACAddress.isEmpty())
1209 i_generateMACAddress();
1210 mData->mode = NetworkAttachmentType_NAT;
1211 }
1212 mData->fCableConnected = true;
1213}
1214
1215ComObjPtr<NetworkAdapter> NetworkAdapter::i_getPeer()
1216{
1217 return mPeer;
1218}
1219
1220
1221// private methods
1222////////////////////////////////////////////////////////////////////////////////
1223
1224/**
1225 * Generates a new unique MAC address based on our vendor ID and
1226 * parts of a GUID.
1227 *
1228 * @note Must be called from under the object's write lock or within the init
1229 * span.
1230 */
1231void NetworkAdapter::i_generateMACAddress()
1232{
1233 Utf8Str mac;
1234 Host::i_generateMACAddress(mac);
1235 LogFlowThisFunc(("generated MAC: '%s'\n", mac.c_str()));
1236 mData->strMACAddress = mac;
1237}
1238
1239HRESULT NetworkAdapter::getBandwidthGroup(ComPtr<IBandwidthGroup> &aBandwidthGroup)
1240{
1241 LogFlowThisFuncEnter();
1242
1243 HRESULT hrc = S_OK;
1244
1245 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1246
1247 if (mData->strBandwidthGroup.isNotEmpty())
1248 {
1249 ComObjPtr<BandwidthGroup> pBwGroup;
1250 hrc = mParent->i_getBandwidthGroup(mData->strBandwidthGroup, pBwGroup, true /* fSetError */);
1251
1252 Assert(SUCCEEDED(hrc)); /* This is not allowed to fail because the existence
1253 * of the group was checked when it was attached. */
1254 if (SUCCEEDED(hrc))
1255 pBwGroup.queryInterfaceTo(aBandwidthGroup.asOutParam());
1256 }
1257
1258 LogFlowThisFuncLeave();
1259 return hrc;
1260}
1261
1262HRESULT NetworkAdapter::setBandwidthGroup(const ComPtr<IBandwidthGroup> &aBandwidthGroup)
1263{
1264 LogFlowThisFuncEnter();
1265
1266 /* the machine needs to be mutable */
1267 AutoMutableOrSavedStateDependency adep(mParent);
1268 if (FAILED(adep.rc())) return adep.rc();
1269
1270 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1271
1272 IBandwidthGroup *iBw = aBandwidthGroup;
1273 Utf8Str strBwGroup;
1274 if (aBandwidthGroup)
1275 strBwGroup = static_cast<BandwidthGroup *>(iBw)->i_getName();
1276
1277 if (mData->strBandwidthGroup != strBwGroup)
1278 {
1279 ComObjPtr<BandwidthGroup> pBwGroup;
1280 if (!strBwGroup.isEmpty())
1281 {
1282 HRESULT hrc = mParent->i_getBandwidthGroup(strBwGroup, pBwGroup, false /* fSetError */);
1283 NOREF(hrc);
1284 Assert(SUCCEEDED(hrc)); /* This is not allowed to fail because the existence
1285 of the group was checked when it was attached. */
1286 }
1287
1288 i_updateBandwidthGroup(pBwGroup);
1289
1290 // leave the lock before informing callbacks
1291 alock.release();
1292
1293 AutoWriteLock mlock(mParent COMMA_LOCKVAL_SRC_POS);
1294 mParent->i_setModified(Machine::IsModified_NetworkAdapters);
1295 mlock.release();
1296
1297 /* TODO: changeAdapter=???. */
1298 mParent->i_onNetworkAdapterChange(this, FALSE);
1299 }
1300
1301 LogFlowThisFuncLeave();
1302 return S_OK;
1303}
1304
1305void NetworkAdapter::i_updateBandwidthGroup(BandwidthGroup *aBwGroup)
1306{
1307 LogFlowThisFuncEnter();
1308 Assert(isWriteLockOnCurrentThread());
1309
1310 ComObjPtr<BandwidthGroup> pOldBwGroup;
1311 if (!mData->strBandwidthGroup.isEmpty())
1312 {
1313 HRESULT hrc = mParent->i_getBandwidthGroup(mData->strBandwidthGroup, pOldBwGroup, false /* fSetError */);
1314 NOREF(hrc);
1315 Assert(SUCCEEDED(hrc)); /* This is not allowed to fail because the existence of
1316 the group was checked when it was attached. */
1317 }
1318
1319 mData.backup();
1320 if (!pOldBwGroup.isNull())
1321 {
1322 pOldBwGroup->i_release();
1323 mData->strBandwidthGroup = Utf8Str::Empty;
1324 }
1325
1326 if (aBwGroup)
1327 {
1328 mData->strBandwidthGroup = aBwGroup->i_getName();
1329 aBwGroup->i_reference();
1330 }
1331
1332 LogFlowThisFuncLeave();
1333}
1334
1335
1336HRESULT NetworkAdapter::i_checkAndSwitchFromNatNetworking(com::Utf8Str networkName)
1337{
1338 HRESULT hrc;
1339 MachineState_T state;
1340
1341 hrc = mParent->COMGETTER(State)(&state);
1342 if (FAILED(hrc))
1343 return hrc;
1344
1345 if ( state == MachineState_Running
1346 || state == MachineState_Paused)
1347 {
1348 Bstr bstrName;
1349 hrc = mParent->COMGETTER(Name)(bstrName.asOutParam());
1350 LogRel(("VM '%ls' stops using NAT network '%s'\n", bstrName.raw(), networkName.c_str()));
1351 int natCount = mParent->i_getVirtualBox()->i_natNetworkRefDec(Bstr(networkName).raw());
1352 if (natCount == -1)
1353 return E_INVALIDARG; /* no such network */
1354 }
1355
1356 return S_OK;
1357}
1358
1359
1360HRESULT NetworkAdapter::i_switchToNatNetworking(const com::Utf8Str &aNatNetworkName)
1361{
1362 HRESULT hrc;
1363 MachineState_T state;
1364
1365 hrc = mParent->COMGETTER(State)(&state);
1366 if (FAILED(hrc))
1367 return hrc;
1368
1369 if ( state == MachineState_Running
1370 || state == MachineState_Paused)
1371 {
1372 Bstr bstrName;
1373 hrc = mParent->COMGETTER(Name)(bstrName.asOutParam());
1374 LogRel(("VM '%ls' starts using NAT network '%s'\n", bstrName.raw(), aNatNetworkName.c_str()));
1375 int natCount = mParent->i_getVirtualBox()->i_natNetworkRefInc(Bstr(aNatNetworkName).raw());
1376 if (natCount == -1)
1377 return E_INVALIDARG; /* not found */
1378 }
1379
1380 return S_OK;
1381}
1382/* 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