NetworkInterface.cs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713
  1. //
  2. // System.Net.NetworkInformation.NetworkInterface
  3. //
  4. // Authors:
  5. // Gonzalo Paniagua Javier ([email protected])
  6. // Atsushi Enomoto ([email protected])
  7. // Miguel de Icaza ([email protected])
  8. // Eric Butler ([email protected])
  9. // Marek Habersack ([email protected])
  10. //
  11. // Copyright (c) 2006-2008 Novell, Inc. (http://www.novell.com)
  12. //
  13. // Permission is hereby granted, free of charge, to any person obtaining
  14. // a copy of this software and associated documentation files (the
  15. // "Software"), to deal in the Software without restriction, including
  16. // without limitation the rights to use, copy, modify, merge, publish,
  17. // distribute, sublicense, and/or sell copies of the Software, and to
  18. // permit persons to whom the Software is furnished to do so, subject to
  19. // the following conditions:
  20. //
  21. // The above copyright notice and this permission notice shall be
  22. // included in all copies or substantial portions of the Software.
  23. //
  24. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  25. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  26. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  27. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  28. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  29. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  30. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  31. //
  32. using System;
  33. using System.Collections.Generic;
  34. using System.Collections;
  35. using System.Net;
  36. using System.Net.Sockets;
  37. using System.Runtime.InteropServices;
  38. using System.Text;
  39. using System.IO;
  40. using System.Globalization;
  41. namespace System.Net.NetworkInformation {
  42. public abstract class NetworkInterface {
  43. static Version windowsVer51 = new Version (5, 1);
  44. static internal readonly bool runningOnUnix = (Environment.OSVersion.Platform == PlatformID.Unix);
  45. protected NetworkInterface ()
  46. {
  47. }
  48. public static NetworkInterface [] GetAllNetworkInterfaces ()
  49. {
  50. if (runningOnUnix) {
  51. try {
  52. if (Platform.IsMacOS)
  53. return MacOsNetworkInterface.ImplGetAllNetworkInterfaces ();
  54. else
  55. return LinuxNetworkInterface.ImplGetAllNetworkInterfaces ();
  56. } catch (SystemException ex) {
  57. throw ex;
  58. } catch {
  59. return new NetworkInterface [0];
  60. }
  61. } else {
  62. if (Environment.OSVersion.Version >= windowsVer51)
  63. return Win32NetworkInterface2.ImplGetAllNetworkInterfaces ();
  64. return new NetworkInterface [0];
  65. }
  66. }
  67. [MonoTODO("Always returns true")]
  68. public static bool GetIsNetworkAvailable ()
  69. {
  70. return true;
  71. }
  72. internal static string ReadLine (string path)
  73. {
  74. using (FileStream fs = File.OpenRead (path)){
  75. using (StreamReader sr = new StreamReader (fs)){
  76. return sr.ReadLine ();
  77. }
  78. }
  79. }
  80. [MonoTODO("Only works on Linux. Returns 0 on other systems.")]
  81. public static int LoopbackInterfaceIndex {
  82. get {
  83. if (runningOnUnix) {
  84. try {
  85. return UnixNetworkInterface.IfNameToIndex ("lo");
  86. } catch {
  87. return 0;
  88. }
  89. } else
  90. return 0;
  91. }
  92. }
  93. public abstract IPInterfaceProperties GetIPProperties ();
  94. public abstract IPv4InterfaceStatistics GetIPv4Statistics ();
  95. public abstract PhysicalAddress GetPhysicalAddress ();
  96. public abstract bool Supports (NetworkInterfaceComponent networkInterfaceComponent);
  97. public abstract string Description { get; }
  98. public abstract string Id { get; }
  99. public abstract bool IsReceiveOnly { get; }
  100. public abstract string Name { get; }
  101. public abstract NetworkInterfaceType NetworkInterfaceType { get; }
  102. public abstract OperationalStatus OperationalStatus { get; }
  103. public abstract long Speed { get; }
  104. public abstract bool SupportsMulticast { get; }
  105. }
  106. abstract class UnixNetworkInterface : NetworkInterface
  107. {
  108. [DllImport("libc")]
  109. static extern int if_nametoindex(string ifname);
  110. protected IPv4InterfaceStatistics ipv4stats;
  111. protected IPInterfaceProperties ipproperties;
  112. string name;
  113. //int index;
  114. protected List <IPAddress> addresses;
  115. byte[] macAddress;
  116. NetworkInterfaceType type;
  117. internal UnixNetworkInterface (string name)
  118. {
  119. this.name = name;
  120. addresses = new List<IPAddress> ();
  121. }
  122. public static int IfNameToIndex (string ifname)
  123. {
  124. return if_nametoindex(ifname);
  125. }
  126. internal void AddAddress (IPAddress address)
  127. {
  128. addresses.Add (address);
  129. }
  130. internal void SetLinkLayerInfo (int index, byte[] macAddress, NetworkInterfaceType type)
  131. {
  132. //this.index = index;
  133. this.macAddress = macAddress;
  134. this.type = type;
  135. }
  136. public override PhysicalAddress GetPhysicalAddress ()
  137. {
  138. if (macAddress != null)
  139. return new PhysicalAddress (macAddress);
  140. else
  141. return PhysicalAddress.None;
  142. }
  143. public override bool Supports (NetworkInterfaceComponent networkInterfaceComponent)
  144. {
  145. bool wantIPv4 = networkInterfaceComponent == NetworkInterfaceComponent.IPv4;
  146. bool wantIPv6 = wantIPv4 ? false : networkInterfaceComponent == NetworkInterfaceComponent.IPv6;
  147. foreach (IPAddress address in addresses) {
  148. if (wantIPv4 && address.AddressFamily == AddressFamily.InterNetwork)
  149. return true;
  150. else if (wantIPv6 && address.AddressFamily == AddressFamily.InterNetworkV6)
  151. return true;
  152. }
  153. return false;
  154. }
  155. public override string Description {
  156. get { return name; }
  157. }
  158. public override string Id {
  159. get { return name; }
  160. }
  161. public override bool IsReceiveOnly {
  162. get { return false; }
  163. }
  164. public override string Name {
  165. get { return name; }
  166. }
  167. public override NetworkInterfaceType NetworkInterfaceType {
  168. get { return type; }
  169. }
  170. [MonoTODO ("Parse dmesg?")]
  171. public override long Speed {
  172. get {
  173. // Bits/s
  174. return 1000000;
  175. }
  176. }
  177. }
  178. //
  179. // This class needs support from the libsupport.so library to fetch the
  180. // data using arch-specific ioctls.
  181. //
  182. // For this to work, we have to create this on the factory above.
  183. //
  184. class LinuxNetworkInterface : UnixNetworkInterface
  185. {
  186. [DllImport ("libc")]
  187. static extern int getifaddrs (out IntPtr ifap);
  188. [DllImport ("libc")]
  189. static extern void freeifaddrs (IntPtr ifap);
  190. const int AF_INET = 2;
  191. const int AF_INET6 = 10;
  192. const int AF_PACKET = 17;
  193. //NetworkInterfaceType type;
  194. string iface_path;
  195. string iface_operstate_path;
  196. string iface_flags_path;
  197. internal string IfacePath {
  198. get { return iface_path; }
  199. }
  200. public static NetworkInterface [] ImplGetAllNetworkInterfaces ()
  201. {
  202. var interfaces = new Dictionary <string, LinuxNetworkInterface> ();
  203. IntPtr ifap;
  204. if (getifaddrs (out ifap) != 0)
  205. throw new SystemException ("getifaddrs() failed");
  206. try {
  207. IntPtr next = ifap;
  208. while (next != IntPtr.Zero) {
  209. ifaddrs addr = (ifaddrs) Marshal.PtrToStructure (next, typeof (ifaddrs));
  210. IPAddress address = IPAddress.None;
  211. string name = addr.ifa_name;
  212. int index = -1;
  213. byte[] macAddress = null;
  214. NetworkInterfaceType type = NetworkInterfaceType.Unknown;
  215. if (addr.ifa_addr != IntPtr.Zero) {
  216. sockaddr_in sockaddr = (sockaddr_in) Marshal.PtrToStructure (addr.ifa_addr, typeof (sockaddr_in));
  217. if (sockaddr.sin_family == AF_INET6) {
  218. sockaddr_in6 sockaddr6 = (sockaddr_in6) Marshal.PtrToStructure (addr.ifa_addr, typeof (sockaddr_in6));
  219. address = new IPAddress (sockaddr6.sin6_addr.u6_addr8, sockaddr6.sin6_scope_id);
  220. } else if (sockaddr.sin_family == AF_INET) {
  221. address = new IPAddress (sockaddr.sin_addr);
  222. } else if (sockaddr.sin_family == AF_PACKET) {
  223. sockaddr_ll sockaddrll = (sockaddr_ll) Marshal.PtrToStructure (addr.ifa_addr, typeof (sockaddr_ll));
  224. if (((int)sockaddrll.sll_halen) > sockaddrll.sll_addr.Length){
  225. Console.Error.WriteLine ("Got a bad hardware address length for an AF_PACKET {0} {1}",
  226. sockaddrll.sll_halen, sockaddrll.sll_addr.Length);
  227. next = addr.ifa_next;
  228. continue;
  229. }
  230. macAddress = new byte [(int) sockaddrll.sll_halen];
  231. Array.Copy (sockaddrll.sll_addr, 0, macAddress, 0, macAddress.Length);
  232. index = sockaddrll.sll_ifindex;
  233. int hwtype = (int)sockaddrll.sll_hatype;
  234. if (Enum.IsDefined (typeof (LinuxArpHardware), hwtype)) {
  235. switch ((LinuxArpHardware)hwtype) {
  236. case LinuxArpHardware.EETHER:
  237. goto case LinuxArpHardware.ETHER;
  238. case LinuxArpHardware.ETHER:
  239. type = NetworkInterfaceType.Ethernet;
  240. break;
  241. case LinuxArpHardware.PRONET:
  242. type = NetworkInterfaceType.TokenRing;
  243. break;
  244. case LinuxArpHardware.ATM:
  245. type = NetworkInterfaceType.Atm;
  246. break;
  247. case LinuxArpHardware.SLIP:
  248. type = NetworkInterfaceType.Slip;
  249. break;
  250. case LinuxArpHardware.PPP:
  251. type = NetworkInterfaceType.Ppp;
  252. break;
  253. case LinuxArpHardware.LOOPBACK:
  254. type = NetworkInterfaceType.Loopback;
  255. macAddress = null;
  256. break;
  257. case LinuxArpHardware.FDDI:
  258. type = NetworkInterfaceType.Fddi;
  259. break;
  260. case LinuxArpHardware.TUNNEL6:
  261. goto case LinuxArpHardware.TUNNEL;
  262. case LinuxArpHardware.TUNNEL:
  263. type = NetworkInterfaceType.Tunnel;
  264. break;
  265. }
  266. }
  267. }
  268. }
  269. LinuxNetworkInterface iface = null;
  270. if (!interfaces.TryGetValue (name, out iface)) {
  271. iface = new LinuxNetworkInterface (name);
  272. interfaces.Add (name, iface);
  273. }
  274. if (!address.Equals (IPAddress.None))
  275. iface.AddAddress (address);
  276. if (macAddress != null || type == NetworkInterfaceType.Loopback) {
  277. if (type == NetworkInterfaceType.Ethernet) {
  278. if (Directory.Exists(iface.IfacePath + "wireless")) {
  279. type = NetworkInterfaceType.Wireless80211;
  280. }
  281. }
  282. iface.SetLinkLayerInfo (index, macAddress, type);
  283. }
  284. next = addr.ifa_next;
  285. }
  286. } finally {
  287. freeifaddrs (ifap);
  288. }
  289. NetworkInterface [] result = new NetworkInterface [interfaces.Count];
  290. int x = 0;
  291. foreach (NetworkInterface thisInterface in interfaces.Values) {
  292. result [x] = thisInterface;
  293. x++;
  294. }
  295. return result;
  296. }
  297. LinuxNetworkInterface (string name)
  298. : base (name)
  299. {
  300. iface_path = "/sys/class/net/" + name + "/";
  301. iface_operstate_path = iface_path + "operstate";
  302. iface_flags_path = iface_path + "flags";
  303. }
  304. public override IPInterfaceProperties GetIPProperties ()
  305. {
  306. if (ipproperties == null)
  307. ipproperties = new LinuxIPInterfaceProperties (this, addresses);
  308. return ipproperties;
  309. }
  310. public override IPv4InterfaceStatistics GetIPv4Statistics ()
  311. {
  312. if (ipv4stats == null)
  313. ipv4stats = new LinuxIPv4InterfaceStatistics (this);
  314. return ipv4stats;
  315. }
  316. public override OperationalStatus OperationalStatus {
  317. get {
  318. if (!Directory.Exists (iface_path))
  319. return OperationalStatus.Unknown;
  320. try {
  321. string s = ReadLine (iface_operstate_path);
  322. switch (s){
  323. case "unknown":
  324. return OperationalStatus.Unknown;
  325. case "notpresent":
  326. return OperationalStatus.NotPresent;
  327. case "down":
  328. return OperationalStatus.Down;
  329. case "lowerlayerdown":
  330. return OperationalStatus.LowerLayerDown;
  331. case "testing":
  332. return OperationalStatus.Testing;
  333. case "dormant":
  334. return OperationalStatus.Dormant;
  335. case "up":
  336. return OperationalStatus.Up;
  337. }
  338. } catch {
  339. }
  340. return OperationalStatus.Unknown;
  341. }
  342. }
  343. public override bool SupportsMulticast {
  344. get {
  345. if (!Directory.Exists (iface_path))
  346. return false;
  347. try {
  348. string s = ReadLine (iface_flags_path);
  349. if (s.Length > 2 && s [0] == '0' && s [1] == 'x')
  350. s = s.Substring (2);
  351. ulong f = UInt64.Parse (s, NumberStyles.HexNumber);
  352. // Hardcoded, only useful for Linux.
  353. return ((f & 0x1000) == 0x1000);
  354. } catch {
  355. return false;
  356. }
  357. }
  358. }
  359. }
  360. class MacOsNetworkInterface : UnixNetworkInterface
  361. {
  362. [DllImport ("libc")]
  363. static extern int getifaddrs (out IntPtr ifap);
  364. [DllImport ("libc")]
  365. static extern void freeifaddrs (IntPtr ifap);
  366. const int AF_INET = 2;
  367. const int AF_INET6 = 30;
  368. const int AF_LINK = 18;
  369. public static NetworkInterface [] ImplGetAllNetworkInterfaces ()
  370. {
  371. var interfaces = new Dictionary <string, MacOsNetworkInterface> ();
  372. IntPtr ifap;
  373. if (getifaddrs (out ifap) != 0)
  374. throw new SystemException ("getifaddrs() failed");
  375. try {
  376. IntPtr next = ifap;
  377. while (next != IntPtr.Zero) {
  378. MacOsStructs.ifaddrs addr = (MacOsStructs.ifaddrs) Marshal.PtrToStructure (next, typeof (MacOsStructs.ifaddrs));
  379. IPAddress address = IPAddress.None;
  380. string name = addr.ifa_name;
  381. int index = -1;
  382. byte[] macAddress = null;
  383. NetworkInterfaceType type = NetworkInterfaceType.Unknown;
  384. if (addr.ifa_addr != IntPtr.Zero) {
  385. MacOsStructs.sockaddr sockaddr = (MacOsStructs.sockaddr) Marshal.PtrToStructure (addr.ifa_addr, typeof (MacOsStructs.sockaddr));
  386. if (sockaddr.sa_family == AF_INET6) {
  387. MacOsStructs.sockaddr_in6 sockaddr6 = (MacOsStructs.sockaddr_in6) Marshal.PtrToStructure (addr.ifa_addr, typeof (MacOsStructs.sockaddr_in6));
  388. address = new IPAddress (sockaddr6.sin6_addr.u6_addr8, sockaddr6.sin6_scope_id);
  389. } else if (sockaddr.sa_family == AF_INET) {
  390. MacOsStructs.sockaddr_in sockaddrin = (MacOsStructs.sockaddr_in) Marshal.PtrToStructure (addr.ifa_addr, typeof (MacOsStructs.sockaddr_in));
  391. address = new IPAddress (sockaddrin.sin_addr);
  392. } else if (sockaddr.sa_family == AF_LINK) {
  393. MacOsStructs.sockaddr_dl sockaddrdl = (MacOsStructs.sockaddr_dl) Marshal.PtrToStructure (addr.ifa_addr, typeof (MacOsStructs.sockaddr_dl));
  394. macAddress = new byte [(int) sockaddrdl.sdl_alen];
  395. Array.Copy (sockaddrdl.sdl_data, sockaddrdl.sdl_nlen, macAddress, 0, Math.Min (macAddress.Length, sockaddrdl.sdl_data.Length - sockaddrdl.sdl_nlen));
  396. index = sockaddrdl.sdl_index;
  397. int hwtype = (int) sockaddrdl.sdl_type;
  398. if (Enum.IsDefined (typeof (MacOsArpHardware), hwtype)) {
  399. switch ((MacOsArpHardware) hwtype) {
  400. case MacOsArpHardware.ETHER:
  401. type = NetworkInterfaceType.Ethernet;
  402. break;
  403. case MacOsArpHardware.ATM:
  404. type = NetworkInterfaceType.Atm;
  405. break;
  406. case MacOsArpHardware.SLIP:
  407. type = NetworkInterfaceType.Slip;
  408. break;
  409. case MacOsArpHardware.PPP:
  410. type = NetworkInterfaceType.Ppp;
  411. break;
  412. case MacOsArpHardware.LOOPBACK:
  413. type = NetworkInterfaceType.Loopback;
  414. macAddress = null;
  415. break;
  416. case MacOsArpHardware.FDDI:
  417. type = NetworkInterfaceType.Fddi;
  418. break;
  419. }
  420. }
  421. }
  422. }
  423. MacOsNetworkInterface iface = null;
  424. if (!interfaces.TryGetValue (name, out iface)) {
  425. iface = new MacOsNetworkInterface (name);
  426. interfaces.Add (name, iface);
  427. }
  428. if (!address.Equals (IPAddress.None))
  429. iface.AddAddress (address);
  430. if (macAddress != null || type == NetworkInterfaceType.Loopback)
  431. iface.SetLinkLayerInfo (index, macAddress, type);
  432. next = addr.ifa_next;
  433. }
  434. } finally {
  435. freeifaddrs (ifap);
  436. }
  437. NetworkInterface [] result = new NetworkInterface [interfaces.Count];
  438. int x = 0;
  439. foreach (NetworkInterface thisInterface in interfaces.Values) {
  440. result [x] = thisInterface;
  441. x++;
  442. }
  443. return result;
  444. }
  445. MacOsNetworkInterface (string name)
  446. : base (name)
  447. {
  448. }
  449. public override IPInterfaceProperties GetIPProperties ()
  450. {
  451. if (ipproperties == null)
  452. ipproperties = new MacOsIPInterfaceProperties (this, addresses);
  453. return ipproperties;
  454. }
  455. public override IPv4InterfaceStatistics GetIPv4Statistics ()
  456. {
  457. if (ipv4stats == null)
  458. ipv4stats = new MacOsIPv4InterfaceStatistics (this);
  459. return ipv4stats;
  460. }
  461. public override OperationalStatus OperationalStatus {
  462. get {
  463. return OperationalStatus.Unknown;
  464. }
  465. }
  466. public override bool SupportsMulticast {
  467. get {
  468. return false;
  469. }
  470. }
  471. }
  472. class Win32NetworkInterface2 : NetworkInterface
  473. {
  474. [DllImport ("iphlpapi.dll", SetLastError = true)]
  475. static extern int GetAdaptersInfo (byte [] info, ref int size);
  476. [DllImport ("iphlpapi.dll", SetLastError = true)]
  477. static extern int GetAdaptersAddresses (uint family, uint flags, IntPtr reserved, byte [] info, ref int size);
  478. [DllImport ("iphlpapi.dll", SetLastError = true)]
  479. static extern int GetIfEntry (ref Win32_MIB_IFROW row);
  480. public static NetworkInterface [] ImplGetAllNetworkInterfaces ()
  481. {
  482. // Win32_IP_ADAPTER_INFO [] ai = GetAdaptersInfo ();
  483. Win32_IP_ADAPTER_ADDRESSES [] aa = GetAdaptersAddresses ();
  484. NetworkInterface [] ret = new NetworkInterface [aa.Length];
  485. for (int i = 0; i < ret.Length; i++)
  486. ret [i] = new Win32NetworkInterface2 (aa [i]);
  487. return ret;
  488. }
  489. public static Win32_IP_ADAPTER_INFO GetAdapterInfoByIndex (int index)
  490. {
  491. foreach (Win32_IP_ADAPTER_INFO info in GetAdaptersInfo ())
  492. if (info.Index == index)
  493. return info;
  494. return null;
  495. }
  496. unsafe static Win32_IP_ADAPTER_INFO [] GetAdaptersInfo ()
  497. {
  498. byte [] bytes = null;
  499. int len = 0;
  500. GetAdaptersInfo (bytes, ref len);
  501. bytes = new byte [len];
  502. int ret = GetAdaptersInfo (bytes, ref len);
  503. if (ret != 0)
  504. throw new NetworkInformationException (ret);
  505. List<Win32_IP_ADAPTER_INFO> l = new List<Win32_IP_ADAPTER_INFO> ();
  506. fixed (byte* ptr = bytes) {
  507. Win32_IP_ADAPTER_INFO info;
  508. for (IntPtr p = (IntPtr) ptr; p != IntPtr.Zero; p = info.Next) {
  509. info = new Win32_IP_ADAPTER_INFO ();
  510. Marshal.PtrToStructure (p, info);
  511. l.Add (info);
  512. }
  513. }
  514. return l.ToArray ();
  515. }
  516. unsafe static Win32_IP_ADAPTER_ADDRESSES [] GetAdaptersAddresses ()
  517. {
  518. byte [] bytes = null;
  519. int len = 0;
  520. GetAdaptersAddresses (0, 0, IntPtr.Zero, bytes, ref len);
  521. bytes = new byte [len];
  522. int ret = GetAdaptersAddresses (0, 0, IntPtr.Zero, bytes, ref len);
  523. if (ret != 0)
  524. throw new NetworkInformationException (ret);
  525. List<Win32_IP_ADAPTER_ADDRESSES> l = new List<Win32_IP_ADAPTER_ADDRESSES> ();
  526. fixed (byte* ptr = bytes) {
  527. Win32_IP_ADAPTER_ADDRESSES info;
  528. for (IntPtr p = (IntPtr) ptr; p != IntPtr.Zero; p = info.Next) {
  529. info = new Win32_IP_ADAPTER_ADDRESSES ();
  530. Marshal.PtrToStructure (p, info);
  531. l.Add (info);
  532. }
  533. }
  534. return l.ToArray ();
  535. }
  536. Win32_IP_ADAPTER_ADDRESSES addr;
  537. Win32_MIB_IFROW mib4, mib6;
  538. Win32IPv4InterfaceStatistics ip4stats;
  539. IPInterfaceProperties ip_if_props;
  540. Win32NetworkInterface2 (Win32_IP_ADAPTER_ADDRESSES addr)
  541. {
  542. this.addr = addr;
  543. mib4 = default (Win32_MIB_IFROW);
  544. mib4.Index = addr.Alignment.IfIndex;
  545. if (GetIfEntry (ref mib4) != 0)
  546. mib4.Index = -1; // unavailable;
  547. mib6 = default (Win32_MIB_IFROW);
  548. mib6.Index = addr.Ipv6IfIndex;
  549. if (GetIfEntry (ref mib6) != 0)
  550. mib6.Index = -1; // unavailable;
  551. ip4stats = new Win32IPv4InterfaceStatistics (mib4);
  552. ip_if_props = new Win32IPInterfaceProperties2 (addr, mib4, mib6);
  553. }
  554. public override IPInterfaceProperties GetIPProperties ()
  555. {
  556. return ip_if_props;
  557. }
  558. public override IPv4InterfaceStatistics GetIPv4Statistics ()
  559. {
  560. return ip4stats;
  561. }
  562. public override PhysicalAddress GetPhysicalAddress ()
  563. {
  564. byte [] bytes = new byte [addr.PhysicalAddressLength];
  565. Array.Copy (addr.PhysicalAddress, 0, bytes, 0, bytes.Length);
  566. return new PhysicalAddress (bytes);
  567. }
  568. public override bool Supports (NetworkInterfaceComponent networkInterfaceComponent)
  569. {
  570. switch (networkInterfaceComponent) {
  571. case NetworkInterfaceComponent.IPv4:
  572. return mib4.Index >= 0;
  573. case NetworkInterfaceComponent.IPv6:
  574. return mib6.Index >= 0;
  575. }
  576. return false;
  577. }
  578. public override string Description {
  579. get { return addr.Description; }
  580. }
  581. public override string Id {
  582. get { return addr.AdapterName; }
  583. }
  584. public override bool IsReceiveOnly {
  585. get { return addr.IsReceiveOnly; }
  586. }
  587. public override string Name {
  588. get { return addr.FriendlyName; }
  589. }
  590. public override NetworkInterfaceType NetworkInterfaceType {
  591. get { return addr.IfType; }
  592. }
  593. public override OperationalStatus OperationalStatus {
  594. get { return addr.OperStatus; }
  595. }
  596. public override long Speed {
  597. get { return mib6.Index >= 0 ? mib6.Speed : mib4.Speed; }
  598. }
  599. public override bool SupportsMulticast {
  600. get { return !addr.NoMulticast; }
  601. }
  602. }
  603. }