| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | // | 
					
						
							|  |  |  | // System.Net.NetworkInformation.Ping | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Authors: | 
					
						
							|  |  |  | //	Gonzalo Paniagua Javier (gonzalo@novell.com) | 
					
						
							|  |  |  | //	Atsushi Enomoto (atsushi@ximian.com) | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Copyright (c) 2006-2007 Novell, Inc. (http://www.novell.com) | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | // Copyright 2015 Xamarin Inc. | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | // | 
					
						
							|  |  |  | // 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. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | using System; | 
					
						
							|  |  |  | using System.IO; | 
					
						
							|  |  |  | using System.Text; | 
					
						
							|  |  |  | using System.Diagnostics; | 
					
						
							|  |  |  | using System.Globalization; | 
					
						
							|  |  |  | using System.ComponentModel; | 
					
						
							|  |  |  | using System.Net.Sockets; | 
					
						
							|  |  |  | using System.Security.Principal; | 
					
						
							|  |  |  | using System.Security.Cryptography; | 
					
						
							|  |  |  | using System.Runtime.InteropServices; | 
					
						
							|  |  |  | using System.Threading; | 
					
						
							|  |  |  | using System.Threading.Tasks; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace System.Net.NetworkInformation { | 
					
						
							|  |  |  | 	[MonoTODO ("IPv6 support is missing")] | 
					
						
							|  |  |  | 	public class Ping : Component, IDisposable | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #if !MONOTOUCH | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		[StructLayout(LayoutKind.Sequential)] | 
					
						
							|  |  |  | 		struct cap_user_header_t | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			public UInt32 version; | 
					
						
							|  |  |  | 			public Int32 pid; | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		[StructLayout(LayoutKind.Sequential)] | 
					
						
							|  |  |  | 		struct cap_user_data_t | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			public UInt32 effective; | 
					
						
							|  |  |  | 			public UInt32 permitted; | 
					
						
							|  |  |  | 			public UInt32 inheritable; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		const int DefaultCount = 1; | 
					
						
							|  |  |  | 		static readonly string [] PingBinPaths = new string [] { | 
					
						
							|  |  |  | 			"/bin/ping", | 
					
						
							|  |  |  | 			"/sbin/ping", | 
					
						
							|  |  |  | 			"/usr/sbin/ping", | 
					
						
							|  |  |  | #if MONODROID | 
					
						
							|  |  |  | 			"/system/bin/ping" | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 		static readonly string PingBinPath; | 
					
						
							| 
									
										
										
										
											2017-04-10 11:41:01 +00:00
										 |  |  | 		static bool canSendPrivileged; | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #endif | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		const int default_timeout = 4000; // 4 sec. | 
					
						
							|  |  |  | 		ushort identifier; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 		// Request 32-bit capabilities by using version 1 | 
					
						
							|  |  |  | 		const UInt32 _LINUX_CAPABILITY_VERSION_1 = 0x19980330; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		static readonly byte [] default_buffer = new byte [0]; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		BackgroundWorker worker; | 
					
						
							|  |  |  | 		object user_async_state; | 
					
						
							|  |  |  | 		CancellationTokenSource cts; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		public event PingCompletedEventHandler PingCompleted; | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 13:16:24 +00:00
										 |  |  | #if !MONOTOUCH && !ORBIS | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		static Ping () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (Environment.OSVersion.Platform == PlatformID.Unix) { | 
					
						
							|  |  |  | 				CheckLinuxCapabilities (); | 
					
						
							|  |  |  | 				if (!canSendPrivileged && WindowsIdentity.GetCurrent ().Name == "root") | 
					
						
							|  |  |  | 					canSendPrivileged = true; | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 				// Since different Unix systems can have different path to bin, we try some | 
					
						
							|  |  |  | 				// of the known ones. | 
					
						
							|  |  |  | 				foreach (string ping_path in PingBinPaths) | 
					
						
							|  |  |  | 					if (File.Exists (ping_path)) { | 
					
						
							|  |  |  | 						PingBinPath = ping_path; | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				canSendPrivileged = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (PingBinPath == null) | 
					
						
							|  |  |  | 				PingBinPath = "/bin/ping"; // default, fallback value | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #endif | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		public Ping () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			// Generate a new random 16 bit identifier for every ping | 
					
						
							|  |  |  | 			RNGCryptoServiceProvider rng = new RNGCryptoServiceProvider (); | 
					
						
							|  |  |  | 			byte [] randomIdentifier = new byte [2]; | 
					
						
							|  |  |  | 			rng.GetBytes (randomIdentifier); | 
					
						
							|  |  |  | 			identifier = (ushort)(randomIdentifier [0] + (randomIdentifier [1] << 8)); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 13:16:24 +00:00
										 |  |  | #if !MONOTOUCH && !ORBIS | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		[DllImport ("libc", EntryPoint="capget")] | 
					
						
							|  |  |  | 		static extern int capget (ref cap_user_header_t header, ref cap_user_data_t data); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		static void CheckLinuxCapabilities () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			try { | 
					
						
							|  |  |  | 				cap_user_header_t header = new cap_user_header_t (); | 
					
						
							|  |  |  | 				cap_user_data_t data = new cap_user_data_t (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 				header.version = _LINUX_CAPABILITY_VERSION_1; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				int ret = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				try { | 
					
						
							|  |  |  | 					ret = capget (ref header, ref data); | 
					
						
							|  |  |  | 				} catch (Exception) { | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (ret == -1) | 
					
						
							|  |  |  | 					return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				canSendPrivileged = (data.effective & (1 << 13)) != 0; | 
					
						
							|  |  |  | 			} catch { | 
					
						
							|  |  |  | 				canSendPrivileged = false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #endif | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		void IDisposable.Dispose () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		protected void OnPingCompleted (PingCompletedEventArgs e) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			user_async_state = null; | 
					
						
							|  |  |  | 			worker = null; | 
					
						
							| 
									
										
										
										
											2016-08-03 10:59:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (cts != null) { | 
					
						
							|  |  |  | 				cts.Dispose(); | 
					
						
							|  |  |  | 				cts = null; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if (PingCompleted != null) | 
					
						
							|  |  |  | 				PingCompleted (this, e); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Sync | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public PingReply Send (IPAddress address) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return Send (address, default_timeout); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public PingReply Send (IPAddress address, int timeout) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return Send (address, timeout, default_buffer); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public PingReply Send (IPAddress address, int timeout, byte [] buffer) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return Send (address, timeout, buffer, new PingOptions ()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public PingReply Send (string hostNameOrAddress) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return Send (hostNameOrAddress, default_timeout); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public PingReply Send (string hostNameOrAddress, int timeout) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return Send (hostNameOrAddress, timeout, default_buffer); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public PingReply Send (string hostNameOrAddress, int timeout, byte [] buffer) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return Send (hostNameOrAddress, timeout, buffer, new PingOptions ()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public PingReply Send (string hostNameOrAddress, int timeout, byte [] buffer, PingOptions options) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			IPAddress [] addresses = Dns.GetHostAddresses (hostNameOrAddress); | 
					
						
							|  |  |  | 			return Send (addresses [0], timeout, buffer, options); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public PingReply Send (IPAddress address, int timeout, byte [] buffer, PingOptions options) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (address == null) | 
					
						
							|  |  |  | 				throw new ArgumentNullException ("address"); | 
					
						
							|  |  |  | 			if (timeout < 0) | 
					
						
							|  |  |  | 				throw new ArgumentOutOfRangeException ("timeout", "timeout must be non-negative integer"); | 
					
						
							|  |  |  | 			if (buffer == null) | 
					
						
							|  |  |  | 				throw new ArgumentNullException ("buffer"); | 
					
						
							|  |  |  | 			if (buffer.Length > 65500) | 
					
						
							|  |  |  | 				throw new ArgumentException ("buffer"); | 
					
						
							|  |  |  | 			// options can be null. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #if MONOTOUCH | 
					
						
							|  |  |  | 			throw new InvalidOperationException (); | 
					
						
							|  |  |  | #else | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			if (canSendPrivileged) | 
					
						
							|  |  |  | 				return SendPrivileged (address, timeout, buffer, options); | 
					
						
							|  |  |  | 			return SendUnprivileged (address, timeout, buffer, options); | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #endif | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #if !MONOTOUCH | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		private PingReply SendPrivileged (IPAddress address, int timeout, byte [] buffer, PingOptions options) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			IPEndPoint target = new IPEndPoint (address, 0); | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			 | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			// FIXME: support IPv6 | 
					
						
							|  |  |  | 			using (Socket s = new Socket (AddressFamily.InterNetwork, SocketType.Raw, ProtocolType.Icmp)) { | 
					
						
							|  |  |  | 				if (options != null) { | 
					
						
							|  |  |  | 					s.DontFragment = options.DontFragment; | 
					
						
							|  |  |  | 					s.Ttl = (short) options.Ttl; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				s.SendTimeout = timeout; | 
					
						
							|  |  |  | 				s.ReceiveTimeout = timeout; | 
					
						
							|  |  |  | 				// not sure why Identifier = 0 is unacceptable ... | 
					
						
							|  |  |  | 				IcmpMessage send = new IcmpMessage (8, 0, identifier, 0, buffer); | 
					
						
							|  |  |  | 				byte [] bytes = send.GetBytes (); | 
					
						
							|  |  |  | 				s.SendBufferSize = bytes.Length; | 
					
						
							|  |  |  | 				s.SendTo (bytes, bytes.Length, SocketFlags.None, target); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 				var sw = Stopwatch.StartNew (); | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				// receive | 
					
						
							|  |  |  | 				bytes = new byte [100]; | 
					
						
							|  |  |  | 				do { | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 					EndPoint endpoint = target; | 
					
						
							| 
									
										
										
										
											2017-04-10 11:41:01 +00:00
										 |  |  | 					SocketError error = 0; | 
					
						
							|  |  |  | 					int rc = s.ReceiveFrom (bytes, 0, 100, SocketFlags.None, | 
					
						
							|  |  |  | 							ref endpoint, out error); | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-10 11:41:01 +00:00
										 |  |  | 					if (error != SocketError.Success) { | 
					
						
							|  |  |  | 						if (error == SocketError.TimedOut) { | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 							return new PingReply (null, new byte [0], options, 0, IPStatus.TimedOut); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						throw new NotSupportedException (String.Format ("Unexpected socket error during ping request: {0}", error)); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 					long rtt = (long) sw.ElapsedMilliseconds; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 					int headerLength = (bytes [0] & 0xF) << 2; | 
					
						
							|  |  |  | 					int bodyLength = rc - headerLength; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// Ping reply to different request. discard it. | 
					
						
							|  |  |  | 					if (!((IPEndPoint) endpoint).Address.Equals (target.Address)) { | 
					
						
							|  |  |  | 						long t = timeout - rtt; | 
					
						
							|  |  |  | 						if (t <= 0) | 
					
						
							|  |  |  | 							return new PingReply (null, new byte [0], options, 0, IPStatus.TimedOut); | 
					
						
							|  |  |  | 						s.ReceiveTimeout = (int) t; | 
					
						
							|  |  |  | 						continue; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					IcmpMessage recv = new IcmpMessage (bytes, headerLength, bodyLength); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					/* discard ping reply to different request or echo requests if running on same host. */ | 
					
						
							|  |  |  | 					if (recv.Identifier != identifier || recv.Type == 8) { | 
					
						
							|  |  |  | 						long t = timeout - rtt; | 
					
						
							|  |  |  | 						if (t <= 0) | 
					
						
							|  |  |  | 							return new PingReply (null, new byte [0], options, 0, IPStatus.TimedOut); | 
					
						
							|  |  |  | 						s.ReceiveTimeout = (int) t; | 
					
						
							|  |  |  | 						continue;  | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					return new PingReply (address, recv.Data, options, rtt, recv.IPStatus); | 
					
						
							|  |  |  | 				} while (true); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		private PingReply SendUnprivileged (IPAddress address, int timeout, byte [] buffer, PingOptions options) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #if MONO_FEATURE_PROCESS_START | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 			var sw = Stopwatch.StartNew (); | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			Process ping = new Process (); | 
					
						
							|  |  |  | 			string args = BuildPingArgs (address, timeout, options); | 
					
						
							|  |  |  | 			long trip_time = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ping.StartInfo.FileName = PingBinPath; | 
					
						
							|  |  |  | 			ping.StartInfo.Arguments = args; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ping.StartInfo.CreateNoWindow = true; | 
					
						
							|  |  |  | 			ping.StartInfo.UseShellExecute = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			ping.StartInfo.RedirectStandardOutput = true; | 
					
						
							|  |  |  | 			ping.StartInfo.RedirectStandardError = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 09:35:12 +01:00
										 |  |  | 			IPStatus status = IPStatus.Unknown; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			try { | 
					
						
							|  |  |  | 				ping.Start (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma warning disable 219 | 
					
						
							|  |  |  | 				string stdout = ping.StandardOutput.ReadToEnd (); | 
					
						
							|  |  |  | 				string stderr = ping.StandardError.ReadToEnd (); | 
					
						
							|  |  |  | #pragma warning restore 219 | 
					
						
							|  |  |  | 				 | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 				trip_time = (long) sw.ElapsedMilliseconds; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 				if (!ping.WaitForExit (timeout) || (ping.HasExited && ping.ExitCode == 2)) | 
					
						
							| 
									
										
										
										
											2015-04-07 09:35:12 +01:00
										 |  |  | 					status = IPStatus.TimedOut; | 
					
						
							|  |  |  | 				else if (ping.ExitCode == 0) | 
					
						
							|  |  |  | 					status = IPStatus.Success; | 
					
						
							|  |  |  | 				else if (ping.ExitCode == 1) | 
					
						
							|  |  |  | 					status = IPStatus.TtlExpired; | 
					
						
							|  |  |  | 			} catch { | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			} finally { | 
					
						
							| 
									
										
										
										
											2015-04-07 09:35:12 +01:00
										 |  |  | 				if (!ping.HasExited) | 
					
						
							|  |  |  | 					ping.Kill (); | 
					
						
							|  |  |  | 				ping.Dispose (); | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-07 09:35:12 +01:00
										 |  |  | 			return new PingReply (address, buffer, options, trip_time, status); | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #else | 
					
						
							|  |  |  | 			throw new PlatformNotSupportedException ("Ping is not supported on this platform."); | 
					
						
							|  |  |  | #endif // MONO_FEATURE_PROCESS_START | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #endif // !MONOTOUCH | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Async | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsync (IPAddress address, int timeout, byte [] buffer, object userToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SendAsync (address, default_timeout, default_buffer, new PingOptions (), userToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsync (IPAddress address, int timeout, object userToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SendAsync (address, default_timeout, default_buffer, userToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsync (IPAddress address, object userToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SendAsync (address, default_timeout, userToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsync (string hostNameOrAddress, int timeout, byte [] buffer, object userToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SendAsync (hostNameOrAddress, timeout, buffer, new PingOptions (), userToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsync (string hostNameOrAddress, int timeout, byte [] buffer, PingOptions options, object userToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			IPAddress address = Dns.GetHostEntry (hostNameOrAddress).AddressList [0]; | 
					
						
							|  |  |  | 			SendAsync (address, timeout, buffer, options, userToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsync (string hostNameOrAddress, int timeout, object userToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SendAsync (hostNameOrAddress, timeout, default_buffer, userToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsync (string hostNameOrAddress, object userToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SendAsync (hostNameOrAddress, default_timeout, userToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsync (IPAddress address, int timeout, byte [] buffer, PingOptions options, object userToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if ((worker != null) || (cts != null)) | 
					
						
							|  |  |  | 				throw new InvalidOperationException ("Another SendAsync operation is in progress"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			worker = new BackgroundWorker (); | 
					
						
							|  |  |  | 			worker.DoWork += delegate (object o, DoWorkEventArgs ea) { | 
					
						
							|  |  |  | 				try { | 
					
						
							|  |  |  | 					user_async_state = ea.Argument; | 
					
						
							|  |  |  | 					ea.Result = Send (address, timeout, buffer, options); | 
					
						
							|  |  |  | 				} catch (Exception ex) { | 
					
						
							|  |  |  | 					ea.Result = ex; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			}; | 
					
						
							|  |  |  | 			worker.WorkerSupportsCancellation = true; | 
					
						
							|  |  |  | 			worker.RunWorkerCompleted += delegate (object o, RunWorkerCompletedEventArgs ea) { | 
					
						
							|  |  |  | 				// Note that RunWorkerCompletedEventArgs.UserState cannot be used (LAMESPEC) | 
					
						
							|  |  |  | 				OnPingCompleted (new PingCompletedEventArgs (ea.Error, ea.Cancelled, user_async_state, ea.Result as PingReply)); | 
					
						
							|  |  |  | 			}; | 
					
						
							|  |  |  | 			worker.RunWorkerAsync (userToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// SendAsyncCancel | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SendAsyncCancel () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (cts != null) { | 
					
						
							|  |  |  | 				cts.Cancel (); | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (worker == null) | 
					
						
							|  |  |  | 				throw new InvalidOperationException ("SendAsync operation is not in progress"); | 
					
						
							|  |  |  | 			worker.CancelAsync (); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #if !MONOTOUCH | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 		// ICMP message | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		class IcmpMessage | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			byte [] bytes; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// received | 
					
						
							|  |  |  | 			public IcmpMessage (byte [] bytes, int offset, int size) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				this.bytes = new byte [size]; | 
					
						
							|  |  |  | 				Buffer.BlockCopy (bytes, offset, this.bytes, 0, size); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// to be sent | 
					
						
							|  |  |  | 			public IcmpMessage (byte type, byte code, ushort identifier, ushort sequence, byte [] data) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				bytes = new byte [data.Length + 8]; | 
					
						
							|  |  |  | 				bytes [0] = type; | 
					
						
							|  |  |  | 				bytes [1] = code; | 
					
						
							|  |  |  | 				bytes [4] = (byte) (identifier & 0xFF); | 
					
						
							|  |  |  | 				bytes [5] = (byte) ((int) identifier >> 8); | 
					
						
							|  |  |  | 				bytes [6] = (byte) (sequence & 0xFF); | 
					
						
							|  |  |  | 				bytes [7] = (byte) ((int) sequence >> 8); | 
					
						
							|  |  |  | 				Buffer.BlockCopy (data, 0, bytes, 8, data.Length); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				ushort checksum = ComputeChecksum (bytes); | 
					
						
							|  |  |  | 				bytes [2] = (byte) (checksum & 0xFF); | 
					
						
							|  |  |  | 				bytes [3] = (byte) ((int) checksum >> 8); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			public byte Type { | 
					
						
							|  |  |  | 				get { return bytes [0]; } | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			public byte Code { | 
					
						
							|  |  |  | 				get { return bytes [1]; } | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			public ushort Identifier { | 
					
						
							|  |  |  | 				get { return (ushort) (bytes [4] + (bytes [5] << 8)); } | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			public ushort Sequence { | 
					
						
							|  |  |  | 				get { return (ushort) (bytes [6] + (bytes [7] << 8)); } | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			public byte [] Data { | 
					
						
							|  |  |  | 				get { | 
					
						
							|  |  |  | 					byte [] data = new byte [bytes.Length - 8]; | 
					
						
							|  |  |  | 					Buffer.BlockCopy (bytes, 8, data, 0, data.Length); | 
					
						
							|  |  |  | 					return data; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			public byte [] GetBytes () | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				return bytes; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			static ushort ComputeChecksum (byte [] data) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				uint ret = 0; | 
					
						
							|  |  |  | 				for (int i = 0; i < data.Length; i += 2) { | 
					
						
							|  |  |  | 					ushort us = i + 1 < data.Length ? data [i + 1] : (byte) 0; | 
					
						
							|  |  |  | 					us <<= 8; | 
					
						
							|  |  |  | 					us += data [i]; | 
					
						
							|  |  |  | 					ret += us; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				ret = (ret >> 16) + (ret & 0xFFFF); | 
					
						
							|  |  |  | 				return (ushort) ~ ret; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			public IPStatus IPStatus { | 
					
						
							|  |  |  | 				get { | 
					
						
							|  |  |  | 					switch (Type) { | 
					
						
							|  |  |  | 					case 0: | 
					
						
							|  |  |  | 						return IPStatus.Success; | 
					
						
							|  |  |  | 					case 3: // destination unreacheable | 
					
						
							|  |  |  | 						switch (Code) { | 
					
						
							|  |  |  | 						case 0: | 
					
						
							|  |  |  | 							return IPStatus.DestinationNetworkUnreachable; | 
					
						
							|  |  |  | 						case 1: | 
					
						
							|  |  |  | 							return IPStatus.DestinationHostUnreachable; | 
					
						
							|  |  |  | 						case 2: | 
					
						
							|  |  |  | 							return IPStatus.DestinationProtocolUnreachable; | 
					
						
							|  |  |  | 						case 3: | 
					
						
							|  |  |  | 							return IPStatus.DestinationPortUnreachable; | 
					
						
							|  |  |  | 						case 4: | 
					
						
							|  |  |  | 							return IPStatus.BadOption; // FIXME: likely wrong | 
					
						
							|  |  |  | 						case 5: | 
					
						
							|  |  |  | 							return IPStatus.BadRoute; // not sure if it is correct | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					case 11: | 
					
						
							|  |  |  | 						switch (Code) { | 
					
						
							|  |  |  | 						case 0: | 
					
						
							|  |  |  | 							return IPStatus.TimeExceeded; | 
					
						
							|  |  |  | 						case 1: | 
					
						
							|  |  |  | 							return IPStatus.TtlReassemblyTimeExceeded; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					case 12: | 
					
						
							|  |  |  | 						return IPStatus.ParameterProblem; | 
					
						
							|  |  |  | 					case 4: | 
					
						
							|  |  |  | 						return IPStatus.SourceQuench; | 
					
						
							|  |  |  | 					case 8: | 
					
						
							|  |  |  | 						return IPStatus.Success; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					return IPStatus.Unknown; | 
					
						
							|  |  |  | 					//throw new NotSupportedException (String.Format ("Unexpected pair of ICMP message type and code: type is {0} and code is {1}", Type, Code)); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		private string BuildPingArgs (IPAddress address, int timeout, PingOptions options) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			CultureInfo culture = CultureInfo.InvariantCulture; | 
					
						
							|  |  |  | 			StringBuilder args = new StringBuilder (); | 
					
						
							|  |  |  | 			uint t = Convert.ToUInt32 (Math.Floor ((timeout + 1000) / 1000.0)); | 
					
						
							| 
									
										
										
										
											2015-04-07 09:35:12 +01:00
										 |  |  | 			bool is_mac = Platform.IsMacOS; | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			if (!is_mac) | 
					
						
							|  |  |  | 				args.AppendFormat (culture, "-q -n -c {0} -w {1} -t {2} -M ", DefaultCount, t, options.Ttl); | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				args.AppendFormat (culture, "-q -n -c {0} -t {1} -o -m {2} ", DefaultCount, t, options.Ttl); | 
					
						
							|  |  |  | 			if (!is_mac) | 
					
						
							|  |  |  | 				args.Append (options.DontFragment ? "do " : "dont "); | 
					
						
							|  |  |  | 			else if (options.DontFragment) | 
					
						
							|  |  |  | 				args.Append ("-D "); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			args.Append (address.ToString ()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return args.ToString (); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-02-22 11:00:01 -05:00
										 |  |  | #endif // !MONOTOUCH | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		public Task<PingReply> SendPingAsync (IPAddress address, int timeout, byte [] buffer) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return SendPingAsync (address, default_timeout, default_buffer, new PingOptions ()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public Task<PingReply> SendPingAsync (IPAddress address, int timeout) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return SendPingAsync (address, default_timeout, default_buffer); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public Task<PingReply> SendPingAsync (IPAddress address) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return SendPingAsync (address, default_timeout); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public Task<PingReply> SendPingAsync (string hostNameOrAddress, int timeout, byte [] buffer) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return SendPingAsync (hostNameOrAddress, timeout, buffer, new PingOptions ()); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public Task<PingReply> SendPingAsync (string hostNameOrAddress, int timeout, byte [] buffer, PingOptions options) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			IPAddress address = Dns.GetHostEntry (hostNameOrAddress).AddressList [0]; | 
					
						
							|  |  |  | 			return SendPingAsync (address, timeout, buffer, options); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public Task<PingReply> SendPingAsync (string hostNameOrAddress, int timeout) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return SendPingAsync (hostNameOrAddress, timeout, default_buffer); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public Task<PingReply> SendPingAsync (string hostNameOrAddress) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return SendPingAsync (hostNameOrAddress, default_timeout); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public Task<PingReply> SendPingAsync (IPAddress address, int timeout, byte [] buffer, PingOptions options) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if ((worker != null) || (cts != null)) | 
					
						
							|  |  |  | 				throw new InvalidOperationException ("Another SendAsync operation is in progress"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-03 10:59:49 +00:00
										 |  |  | 			cts = new CancellationTokenSource(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			var task = Task<PingReply>.Factory.StartNew ( | 
					
						
							|  |  |  | 				() => Send (address, timeout, buffer, options), cts.Token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			task.ContinueWith ((t) => { | 
					
						
							|  |  |  | 				if (t.IsCanceled) | 
					
						
							|  |  |  | 					OnPingCompleted (new PingCompletedEventArgs (null, true, null, null)); | 
					
						
							|  |  |  | 				else if (t.IsFaulted) | 
					
						
							|  |  |  | 					OnPingCompleted (new PingCompletedEventArgs (t.Exception, false, null, null)); | 
					
						
							|  |  |  | 				else | 
					
						
							|  |  |  | 					OnPingCompleted (new PingCompletedEventArgs (null, false, null, t.Result)); | 
					
						
							|  |  |  | 			}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return task; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |