| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | #if SECURITY_DEP | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // AsyncProtocolRequest.cs | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Author: | 
					
						
							|  |  |  | //       Martin Baulig <martin.baulig@xamarin.com> | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Copyright (c) 2015 Xamarin, Inc. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | using System; | 
					
						
							|  |  |  | using System.IO; | 
					
						
							|  |  |  | using System.Net; | 
					
						
							|  |  |  | using System.Net.Security; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | using System.Security.Authentication; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | using SD = System.Diagnostics; | 
					
						
							|  |  |  | using System.Threading; | 
					
						
							|  |  |  | using System.Threading.Tasks; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | using System.Runtime.ExceptionServices; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Mono.Net.Security | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	class BufferOffsetSize | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		public byte[] Buffer; | 
					
						
							|  |  |  | 		public int Offset; | 
					
						
							|  |  |  | 		public int Size; | 
					
						
							|  |  |  | 		public int TotalBytes; | 
					
						
							|  |  |  | 		public bool Complete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public int EndOffset { | 
					
						
							|  |  |  | 			get { return Offset + Size; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public int Remaining { | 
					
						
							|  |  |  | 			get { return Buffer.Length - Offset - Size; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public BufferOffsetSize (byte[] buffer, int offset, int size) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			if (buffer == null) | 
					
						
							|  |  |  | 				throw new ArgumentNullException (nameof (buffer)); | 
					
						
							|  |  |  | 			if (offset < 0) | 
					
						
							|  |  |  | 				throw new ArgumentOutOfRangeException (nameof (offset)); | 
					
						
							|  |  |  | 			if (size < 0 || offset + size > buffer.Length) | 
					
						
							|  |  |  | 				throw new ArgumentOutOfRangeException (nameof (size)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 			Buffer = buffer; | 
					
						
							|  |  |  | 			Offset = offset; | 
					
						
							|  |  |  | 			Size = size; | 
					
						
							|  |  |  | 			Complete = false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override string ToString () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return string.Format ("[BufferOffsetSize: {0} {1}]", Offset, Size); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	class BufferOffsetSize2 : BufferOffsetSize | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		public readonly int InitialSize; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public BufferOffsetSize2 (int size) | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			: base (new byte[size], 0, 0) | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			InitialSize = size; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void Reset () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Offset = Size = 0; | 
					
						
							|  |  |  | 			TotalBytes = 0; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			Buffer = new byte[InitialSize]; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 			Complete = false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void MakeRoom (int size) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (Remaining >= size) | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			int missing = size - Remaining; | 
					
						
							|  |  |  | 			if (Offset == 0 && Size == 0) { | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 				Buffer = new byte[size]; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			var buffer = new byte[Buffer.Length + missing]; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 			Buffer.CopyTo (buffer, 0); | 
					
						
							|  |  |  | 			Buffer = buffer; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void AppendData (byte[] buffer, int offset, int size) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			MakeRoom (size); | 
					
						
							|  |  |  | 			System.Buffer.BlockCopy (buffer, offset, Buffer, EndOffset, size); | 
					
						
							|  |  |  | 			Size += size; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 	enum AsyncOperationStatus | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		Initialize, | 
					
						
							|  |  |  | 		Continue, | 
					
						
							| 
									
										
										
										
											2016-10-12 08:30:39 +00:00
										 |  |  | 		ReadDone, | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		Complete | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 	class AsyncProtocolResult | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		public int UserResult { | 
					
						
							|  |  |  | 			get; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		public ExceptionDispatchInfo Error { | 
					
						
							|  |  |  | 			get; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		public AsyncProtocolResult (int result) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			UserResult = result; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		public AsyncProtocolResult (ExceptionDispatchInfo error) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Error = error; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 	abstract class AsyncProtocolRequest | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		public MobileAuthenticatedStream Parent { | 
					
						
							|  |  |  | 			get; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		public bool RunSynchronously { | 
					
						
							|  |  |  | 			get; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		public int ID => ++next_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public string Name => GetType ().Name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public int UserResult { | 
					
						
							|  |  |  | 			get; | 
					
						
							|  |  |  | 			protected set; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		int Started; | 
					
						
							|  |  |  | 		int RequestedSize; | 
					
						
							|  |  |  | 		int WriteRequested; | 
					
						
							|  |  |  | 		readonly object locker = new object (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		static int next_id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public AsyncProtocolRequest (MobileAuthenticatedStream parent, bool sync) | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			Parent = parent; | 
					
						
							|  |  |  | 			RunSynchronously = sync; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 		[SD.Conditional ("MONO_TLS_DEBUG")] | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		protected void Debug (string message, params object[] args) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			Parent.Debug ("{0}({1}:{2}): {3}", Name, Parent.ID, ID, string.Format (message, args)); | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal void RequestRead (int size) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			lock (locker) { | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 				RequestedSize += size; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 				Debug ("RequestRead: {0}", size); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		internal void RequestWrite () | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			WriteRequested = 1; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		internal async Task<AsyncProtocolResult> StartOperation (CancellationToken cancellationToken) | 
					
						
							| 
									
										
										
										
											2016-10-12 08:30:39 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			Debug ("Start Operation: {0}", this); | 
					
						
							|  |  |  | 			if (Interlocked.CompareExchange (ref Started, 1, 0) != 0) | 
					
						
							|  |  |  | 				throw new InvalidOperationException (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			try { | 
					
						
							|  |  |  | 				await ProcessOperation (cancellationToken).ConfigureAwait (false); | 
					
						
							|  |  |  | 				return new AsyncProtocolResult (UserResult); | 
					
						
							|  |  |  | 			} catch (Exception ex) { | 
					
						
							|  |  |  | 				var info = Parent.SetException (MobileAuthenticatedStream.GetSSPIException (ex)); | 
					
						
							|  |  |  | 				return new AsyncProtocolResult (info); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-10-12 08:30:39 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		async Task ProcessOperation (CancellationToken cancellationToken) | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			var status = AsyncOperationStatus.Initialize; | 
					
						
							|  |  |  | 			while (status != AsyncOperationStatus.Complete) { | 
					
						
							|  |  |  | 				cancellationToken.ThrowIfCancellationRequested (); | 
					
						
							|  |  |  | 				Debug ("ProcessOperation: {0}", status); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				var ret = await InnerRead (cancellationToken).ConfigureAwait (false); | 
					
						
							|  |  |  | 				if (ret != null) { | 
					
						
							|  |  |  | 					if (ret == 0) { | 
					
						
							|  |  |  | 						// End-of-stream | 
					
						
							|  |  |  | 						Debug ("END OF STREAM!"); | 
					
						
							|  |  |  | 						status = AsyncOperationStatus.ReadDone; | 
					
						
							|  |  |  | 					} else if (ret < 0) { | 
					
						
							|  |  |  | 						// remote prematurely closed connection. | 
					
						
							|  |  |  | 						throw new IOException ("Remote prematurely closed connection."); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Debug ("ProcessOperation run: {0}", status); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				AsyncOperationStatus newStatus; | 
					
						
							|  |  |  | 				switch (status) { | 
					
						
							|  |  |  | 				case AsyncOperationStatus.Initialize: | 
					
						
							|  |  |  | 				case AsyncOperationStatus.Continue: | 
					
						
							|  |  |  | 				case AsyncOperationStatus.ReadDone: | 
					
						
							|  |  |  | 					newStatus = Run (status); | 
					
						
							|  |  |  | 					break; | 
					
						
							|  |  |  | 				default: | 
					
						
							|  |  |  | 					throw new InvalidOperationException (); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (Interlocked.Exchange (ref WriteRequested, 0) != 0) { | 
					
						
							|  |  |  | 					// Flush the write queue. | 
					
						
							| 
									
										
										
										
											2017-10-19 20:04:20 +00:00
										 |  |  | 					Debug ("ProcessOperation - flushing write queue"); | 
					
						
							| 
									
										
										
										
											2017-11-28 19:36:51 +00:00
										 |  |  | 					await Parent.InnerWrite (RunSynchronously, cancellationToken).ConfigureAwait (false); | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				Debug ("ProcessOperation done: {0} -> {1}", status, newStatus); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				status = newStatus; | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		async Task<int?> InnerRead (CancellationToken cancellationToken) | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			int? totalRead = null; | 
					
						
							|  |  |  | 			var requestedSize = Interlocked.Exchange (ref RequestedSize, 0); | 
					
						
							|  |  |  | 			while (requestedSize > 0) { | 
					
						
							|  |  |  | 				Debug ("ProcessOperation - read inner: {0}", requestedSize); | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 				var ret = await Parent.InnerRead (RunSynchronously, requestedSize, cancellationToken).ConfigureAwait (false); | 
					
						
							|  |  |  | 				Debug ("ProcessOperation - read inner done: {0} - {1}", requestedSize, ret); | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 				if (ret <= 0) | 
					
						
							|  |  |  | 					return ret; | 
					
						
							|  |  |  | 				if (ret > requestedSize) | 
					
						
							|  |  |  | 					throw new InvalidOperationException (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				totalRead += ret; | 
					
						
							|  |  |  | 				requestedSize -= ret; | 
					
						
							|  |  |  | 				var newRequestedSize = Interlocked.Exchange (ref RequestedSize, 0); | 
					
						
							|  |  |  | 				requestedSize += newRequestedSize; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			return totalRead; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		/* | 
					
						
							|  |  |  | 		 * This will operate on the internal buffers and never block. | 
					
						
							|  |  |  | 		 */ | 
					
						
							|  |  |  | 		protected abstract AsyncOperationStatus Run (AsyncOperationStatus status); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override string ToString () | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			return string.Format ("[{0}]", Name); | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 	class AsyncHandshakeRequest : AsyncProtocolRequest | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		public AsyncHandshakeRequest (MobileAuthenticatedStream parent, bool sync) | 
					
						
							|  |  |  | 			: base (parent, sync) | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		protected override AsyncOperationStatus Run (AsyncOperationStatus status) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return Parent.ProcessHandshake (status); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 	abstract class AsyncReadOrWriteRequest : AsyncProtocolRequest | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		protected BufferOffsetSize UserBuffer { | 
					
						
							|  |  |  | 			get; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		protected int CurrentSize { | 
					
						
							|  |  |  | 			get; set; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-10-12 08:30:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		public AsyncReadOrWriteRequest (MobileAuthenticatedStream parent, bool sync, byte[] buffer, int offset, int size) | 
					
						
							|  |  |  | 			: base (parent, sync) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			UserBuffer = new BufferOffsetSize (buffer, offset, size); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-10-12 08:30:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		public override string ToString () | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return string.Format ("[{0}: {1}]", Name, UserBuffer); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 	class AsyncReadRequest : AsyncReadOrWriteRequest | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		public AsyncReadRequest (MobileAuthenticatedStream parent, bool sync, byte[] buffer, int offset, int size) | 
					
						
							|  |  |  | 			: base (parent, sync, buffer, offset, size) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 		protected override AsyncOperationStatus Run (AsyncOperationStatus status) | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			Debug ("ProcessRead - read user: {0} {1}", this, status); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			var (ret, wantMore) = Parent.ProcessRead (UserBuffer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Debug ("ProcessRead - read user done: {0} - {1} {2}", this, ret, wantMore); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (ret < 0) { | 
					
						
							|  |  |  | 				UserResult = -1; | 
					
						
							|  |  |  | 				return AsyncOperationStatus.Complete; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			CurrentSize += ret; | 
					
						
							|  |  |  | 			UserBuffer.Offset += ret; | 
					
						
							|  |  |  | 			UserBuffer.Size -= ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Debug ("Process Read - read user done #1: {0} - {1} {2}", this, CurrentSize, wantMore); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (wantMore && CurrentSize == 0) | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 				return AsyncOperationStatus.Continue; | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			UserResult = CurrentSize; | 
					
						
							|  |  |  | 			return AsyncOperationStatus.Complete; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	class AsyncWriteRequest : AsyncReadOrWriteRequest | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		public AsyncWriteRequest (MobileAuthenticatedStream parent, bool sync, byte[] buffer, int offset, int size) | 
					
						
							|  |  |  | 			: base (parent, sync, buffer, offset, size) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		protected override AsyncOperationStatus Run (AsyncOperationStatus status) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Debug ("ProcessWrite - write user: {0} {1}", this, status); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (UserBuffer.Size == 0) { | 
					
						
							|  |  |  | 				UserResult = CurrentSize; | 
					
						
							|  |  |  | 				return AsyncOperationStatus.Complete; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			var (ret, wantMore) = Parent.ProcessWrite (UserBuffer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Debug ("ProcessWrite - write user done: {0} - {1} {2}", this, ret, wantMore); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (ret < 0) { | 
					
						
							|  |  |  | 				UserResult = -1; | 
					
						
							| 
									
										
										
										
											2016-10-12 08:30:39 +00:00
										 |  |  | 				return AsyncOperationStatus.Complete; | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 			CurrentSize += ret; | 
					
						
							|  |  |  | 			UserBuffer.Offset += ret; | 
					
						
							|  |  |  | 			UserBuffer.Size -= ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (wantMore) | 
					
						
							|  |  |  | 				return AsyncOperationStatus.Continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			UserResult = CurrentSize; | 
					
						
							|  |  |  | 			return AsyncOperationStatus.Complete; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	class AsyncShutdownRequest : AsyncProtocolRequest | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		public AsyncShutdownRequest (MobileAuthenticatedStream parent) | 
					
						
							|  |  |  | 			: base (parent, false) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		protected override AsyncOperationStatus Run (AsyncOperationStatus status) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return Parent.ProcessShutdown (status); | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-08-21 15:34:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-23 13:20:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | #endif |