VirtualBox

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

Last change on this file since 28087 was 27976, checked in by vboxsync, 15 years ago

*: scm cleans up whitespace and adds a new line at the end of ApplianceimplPrivate.h.

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