VirtualBox

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

Last change on this file since 25903 was 25860, checked in by vboxsync, 15 years ago

Main: cleanup: get rid of VirtualBoxBaseProto, move AutoCaller*/*Span* classes out of VirtualBoxBaseProto class scope and into separate header; move CombinedProgress into separate header (it's only used by Console any more)

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