// // System.Net.NetworkInformation.NetworkInterface // // Authors: // Gonzalo Paniagua Javier (gonzalo@novell.com) // Atsushi Enomoto (atsushi@ximian.com) // Miguel de Icaza (miguel@novell.com) // Eric Butler (eric@extremeboredom.net) // Marek Habersack (mhabersack@novell.com) // Marek Safar (marek.safar@gmail.com) // // Copyright (c) 2006-2008 Novell, Inc. (http://www.novell.com) // // Permission is hereby granted, free of charge, to any person obtaining // a copy of this software and associated documentation files (the // "Software"), to deal in the Software without restriction, including // without limitation the rights to use, copy, modify, merge, publish, // distribute, sublicense, and/or sell copies of the Software, and to // permit persons to whom the Software is furnished to do so, subject to // the following conditions: // // The above copyright notice and this permission notice shall be // included in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // using System.Collections.Generic; using System.Runtime.InteropServices; using System.IO; using System.Globalization; namespace System.Net.NetworkInformation { internal class LinuxNetworkInterfaceAPI : UnixNetworkInterfaceAPI { const int AF_INET = 2; const int AF_INET6 = 10; const int AF_PACKET = 17; static void FreeInterfaceAddresses (IntPtr ifap) { #if MONODROID AndroidPlatform.FreeInterfaceAddresses (ifap); #else freeifaddrs (ifap); #endif } static int GetInterfaceAddresses (out IntPtr ifap) { #if MONODROID return AndroidPlatform.GetInterfaceAddresses (out ifap); #else return getifaddrs (out ifap); #endif } public override NetworkInterface [] GetAllNetworkInterfaces () { var interfaces = new Dictionary (); IntPtr ifap; if (GetInterfaceAddresses (out ifap) != 0) throw new SystemException ("getifaddrs() failed"); try { IntPtr next = ifap; while (next != IntPtr.Zero) { ifaddrs addr = (ifaddrs) Marshal.PtrToStructure (next, typeof (ifaddrs)); IPAddress address = IPAddress.None; string name = addr.ifa_name; int index = -1; byte[] macAddress = null; NetworkInterfaceType type = NetworkInterfaceType.Unknown; int nullNameCount = 0; if (addr.ifa_addr != IntPtr.Zero) { sockaddr_in sockaddr = (sockaddr_in) Marshal.PtrToStructure (addr.ifa_addr, typeof (sockaddr_in)); if (sockaddr.sin_family == AF_INET6) { sockaddr_in6 sockaddr6 = (sockaddr_in6) Marshal.PtrToStructure (addr.ifa_addr, typeof (sockaddr_in6)); address = new IPAddress (sockaddr6.sin6_addr.u6_addr8, sockaddr6.sin6_scope_id); } else if (sockaddr.sin_family == AF_INET) { address = new IPAddress (sockaddr.sin_addr); } else if (sockaddr.sin_family == AF_PACKET) { sockaddr_ll sockaddrll = (sockaddr_ll) Marshal.PtrToStructure (addr.ifa_addr, typeof (sockaddr_ll)); if (((int)sockaddrll.sll_halen) > sockaddrll.sll_addr.Length){ next = addr.ifa_next; continue; } macAddress = new byte [(int) sockaddrll.sll_halen]; Array.Copy (sockaddrll.sll_addr, 0, macAddress, 0, macAddress.Length); index = sockaddrll.sll_ifindex; int hwtype = (int)sockaddrll.sll_hatype; if (Enum.IsDefined (typeof (LinuxArpHardware), hwtype)) { switch ((LinuxArpHardware)hwtype) { case LinuxArpHardware.EETHER: goto case LinuxArpHardware.ETHER; case LinuxArpHardware.ETHER: type = NetworkInterfaceType.Ethernet; break; case LinuxArpHardware.PRONET: type = NetworkInterfaceType.TokenRing; break; case LinuxArpHardware.ATM: type = NetworkInterfaceType.Atm; break; case LinuxArpHardware.SLIP: case LinuxArpHardware.CSLIP: case LinuxArpHardware.SLIP6: case LinuxArpHardware.CSLIP6: type = NetworkInterfaceType.Slip; break; case LinuxArpHardware.PPP: type = NetworkInterfaceType.Ppp; break; case LinuxArpHardware.LOOPBACK: type = NetworkInterfaceType.Loopback; macAddress = null; break; case LinuxArpHardware.FDDI: type = NetworkInterfaceType.Fddi; break; case LinuxArpHardware.SIT: case LinuxArpHardware.IPDDP: case LinuxArpHardware.IPGRE: case LinuxArpHardware.IP6GRE: case LinuxArpHardware.TUNNEL6: case LinuxArpHardware.TUNNEL: type = NetworkInterfaceType.Tunnel; break; } } } } LinuxNetworkInterface iface = null; if (String.IsNullOrEmpty (name)) name = "\0" + (++nullNameCount).ToString (); if (!interfaces.TryGetValue (name, out iface)) { iface = new LinuxNetworkInterface (name); interfaces.Add (name, iface); } if (!address.Equals (IPAddress.None)) iface.AddAddress (address); if (macAddress != null || type == NetworkInterfaceType.Loopback) { if (type == NetworkInterfaceType.Ethernet) { if (Directory.Exists(iface.IfacePath + "wireless")) { type = NetworkInterfaceType.Wireless80211; } } iface.SetLinkLayerInfo (index, macAddress, type); } next = addr.ifa_next; } } finally { FreeInterfaceAddresses (ifap); } NetworkInterface [] result = new NetworkInterface [interfaces.Count]; int x = 0; foreach (NetworkInterface thisInterface in interfaces.Values) { result [x] = thisInterface; x++; } return result; } public override int GetLoopbackInterfaceIndex () { return if_nametoindex ("lo"); } public override IPAddress GetNetMask (IPAddress address) { foreach (ifaddrs networkInteface in GetNetworkInterfaces()) { if (networkInteface.ifa_addr == IntPtr.Zero) continue; var sockaddr = (sockaddr_in)Marshal.PtrToStructure(networkInteface.ifa_addr, typeof(sockaddr_in)); if (sockaddr.sin_family != AF_INET) continue; if (!address.Equals(new IPAddress(sockaddr.sin_addr))) continue; var netmask = (sockaddr_in)Marshal.PtrToStructure(networkInteface.ifa_netmask, typeof(sockaddr_in)); return new IPAddress(netmask.sin_addr); } return null; } private static IEnumerable GetNetworkInterfaces() { IntPtr ifap = IntPtr.Zero; try { if (GetInterfaceAddresses(out ifap) != 0) yield break; var next = ifap; while (next != IntPtr.Zero) { var addr = (ifaddrs)Marshal.PtrToStructure(next, typeof(ifaddrs)); yield return addr; next = addr.ifa_next; } } finally { if (ifap != IntPtr.Zero) FreeInterfaceAddresses(ifap); } } } // // This class needs support from the libsupport.so library to fetch the // data using arch-specific ioctls. // // For this to work, we have to create this on the factory above. // sealed class LinuxNetworkInterface : UnixNetworkInterface { //NetworkInterfaceType type; string iface_path; string iface_operstate_path; string iface_flags_path; #if MONODROID [DllImport ("__Internal")] static extern int _monodroid_get_android_api_level (); [DllImport ("__Internal")] static extern bool _monodroid_get_network_interface_up_state (string ifname, ref bool is_up); [DllImport ("__Internal")] static extern bool _monodroid_get_network_interface_supports_multicast (string ifname, ref bool supports_multicast); bool android_use_java_api; #endif internal string IfacePath { get { return iface_path; } } internal LinuxNetworkInterface (string name) : base (name) { iface_path = "/sys/class/net/" + name + "/"; iface_operstate_path = iface_path + "operstate"; iface_flags_path = iface_path + "flags"; #if MONODROID android_use_java_api = _monodroid_get_android_api_level () >= 24; #endif } public override IPInterfaceProperties GetIPProperties () { if (ipproperties == null) ipproperties = new LinuxIPInterfaceProperties (this, addresses); return ipproperties; } public override IPv4InterfaceStatistics GetIPv4Statistics () { if (ipv4stats == null) ipv4stats = new LinuxIPv4InterfaceStatistics (this); return ipv4stats; } public override OperationalStatus OperationalStatus { get { #if MONODROID if (android_use_java_api) { // Starting from API 24 (Android 7 "Nougat") Android restricts access to many // files in the /sys filesystem (see https://code.google.com/p/android/issues/detail?id=205565 // for more information) and therefore we are forced to call into Java API in // order to get the information. Alas, what we can obtain in this way is quite // limited. In the case of OperationalStatus we can only determine whether the // interface is up or down. There is a way to get more detailed information but // it requires an instance of the Android Context class which is not available // to us here. bool is_up = false; if (_monodroid_get_network_interface_up_state (Name, ref is_up)) return is_up ? OperationalStatus.Up : OperationalStatus.Down; else return OperationalStatus.Unknown; } #endif if (!Directory.Exists (iface_path)) return OperationalStatus.Unknown; try { string s = ReadLine (iface_operstate_path); switch (s){ case "unknown": return OperationalStatus.Unknown; case "notpresent": return OperationalStatus.NotPresent; case "down": return OperationalStatus.Down; case "lowerlayerdown": return OperationalStatus.LowerLayerDown; case "testing": return OperationalStatus.Testing; case "dormant": return OperationalStatus.Dormant; case "up": return OperationalStatus.Up; } } catch { } return OperationalStatus.Unknown; } } public override bool SupportsMulticast { get { #if MONODROID if (android_use_java_api) { // Starting from API 24 (Android 7 "Nougat") Android restricts access to many // files in the /sys filesystem (see https://code.google.com/p/android/issues/detail?id=205565 // for more information) and therefore we are forced to call into Java API in // order to get the information. bool supports_multicast = false; _monodroid_get_network_interface_supports_multicast (Name, ref supports_multicast); return supports_multicast; } #endif if (!Directory.Exists (iface_path)) return false; try { string s = ReadLine (iface_flags_path); if (s.Length > 2 && s [0] == '0' && s [1] == 'x') s = s.Substring (2); ulong f = UInt64.Parse (s, NumberStyles.HexNumber); // Hardcoded, only useful for Linux. return ((f & 0x1000) == 0x1000); } catch { return false; } } } internal static string ReadLine (string path) { using (FileStream fs = File.OpenRead (path)){ using (StreamReader sr = new StreamReader (fs)){ return sr.ReadLine (); } } } } }