VirtualBox

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

Last change on this file since 82411 was 81603, checked in by vboxsync, 5 years ago

Main: Added PCnet-ISA/NE2100/Am79C960 to the API (the device side is long in place).

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