You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			386 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			386 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | // | ||
|  | // 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 <string, LinuxNetworkInterface> (); | ||
|  | 			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<ifaddrs> 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 (); | ||
|  | 				} | ||
|  | 			} | ||
|  | 		} | ||
|  | 	} | ||
|  | } |