VirtualBox

source: vbox/trunk/src/VBox/Main/NetworkAdapterImpl.cpp@ 27147

Last change on this file since 27147 was 26818, checked in by vboxsync, 15 years ago

xtracker 3573 / public #5781: when changing the host adapter or the internal network we are attached to, then immediately change the CFGM attachment logic to make this change effect

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