You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			198 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			198 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //---------------------------------------------------------------- | ||
|  | // Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | //---------------------------------------------------------------- | ||
|  | 
 | ||
|  | namespace System.ServiceModel.Channels | ||
|  | { | ||
|  |     using System; | ||
|  |     using System.Collections.Generic; | ||
|  |     using System.Globalization; | ||
|  |     using System.Net; | ||
|  |     using System.Net.Sockets; | ||
|  |     using System.Runtime; | ||
|  |     using System.Runtime.Diagnostics; | ||
|  |     using System.ServiceModel.Diagnostics; | ||
|  |     using System.Threading; | ||
|  |     using System.Xml; | ||
|  | 
 | ||
|  |     abstract class UdpDuplexChannel : UdpChannelBase<Message>, IDuplexChannel | ||
|  |     {         | ||
|  |         protected UdpDuplexChannel( | ||
|  |             ChannelManagerBase channelMananger,  | ||
|  |             MessageEncoder encoder,  | ||
|  |             BufferManager bufferManager, | ||
|  |             UdpSocket[] sendSockets,  | ||
|  |             UdpRetransmissionSettings retransmissionSettings, | ||
|  |             long maxPendingMessagesTotalSize,  | ||
|  |             EndpointAddress localAddress,  | ||
|  |             Uri via,  | ||
|  |             bool isMulticast,  | ||
|  |             int maxReceivedMessageSize) | ||
|  |             : base(channelMananger, encoder, bufferManager, sendSockets, retransmissionSettings, maxPendingMessagesTotalSize, localAddress, via, isMulticast, maxReceivedMessageSize) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         public virtual EndpointAddress RemoteAddress | ||
|  |         { | ||
|  |             get { return null; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public override T GetProperty<T>() | ||
|  |         { | ||
|  |             if (typeof(T) == typeof(IDuplexChannel)) | ||
|  |             { | ||
|  |                 return (T)(object)this; | ||
|  |             } | ||
|  | 
 | ||
|  |             return base.GetProperty<T>(); | ||
|  |         } | ||
|  | 
 | ||
|  |         public IAsyncResult BeginSend(Message message, AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             return this.BeginSend(message, this.DefaultSendTimeout, callback, state); | ||
|  |         } | ||
|  | 
 | ||
|  |         public IAsyncResult BeginSend(Message message, TimeSpan timeout, AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             ThrowIfDisposedOrNotOpen(); | ||
|  |              | ||
|  |             if (message is NullMessage) | ||
|  |             { | ||
|  |                 return new CompletedAsyncResult(callback, state);  | ||
|  |             } | ||
|  |             AddHeadersTo(message); | ||
|  |             return this.UdpOutputChannel.BeginSend(message, timeout, callback, state); | ||
|  |         } | ||
|  | 
 | ||
|  |         public void EndSend(IAsyncResult result) | ||
|  |         { | ||
|  |             if (result is CompletedAsyncResult) | ||
|  |             { | ||
|  |                 CompletedAsyncResult.End(result);  | ||
|  |             } | ||
|  |             else  | ||
|  |             { | ||
|  |                 this.UdpOutputChannel.EndSend(result); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public void Send(Message message) | ||
|  |         { | ||
|  |             this.Send(message, this.DefaultSendTimeout); | ||
|  |         } | ||
|  | 
 | ||
|  |         public void Send(Message message, TimeSpan timeout) | ||
|  |         { | ||
|  |             if (message is NullMessage) | ||
|  |             { | ||
|  |                 return; | ||
|  |             } | ||
|  | 
 | ||
|  |             this.UdpOutputChannel.Send(message, timeout); | ||
|  |         } | ||
|  | 
 | ||
|  |         public Message Receive() | ||
|  |         { | ||
|  |             return this.Receive(this.DefaultReceiveTimeout); | ||
|  |         } | ||
|  | 
 | ||
|  |         public Message Receive(TimeSpan timeout) | ||
|  |         { | ||
|  |             if (timeout < TimeSpan.Zero) | ||
|  |             { | ||
|  |                 throw FxTrace.Exception.AsError(new ArgumentOutOfRangeException("timeout", timeout, SR.TimeoutOutOfRange0)); | ||
|  |             } | ||
|  | 
 | ||
|  |             this.ThrowPending(); | ||
|  |             return InputChannel.HelpReceive(this, timeout); | ||
|  |         } | ||
|  | 
 | ||
|  |         public IAsyncResult BeginReceive(AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             return this.BeginReceive(this.DefaultReceiveTimeout, callback, state); | ||
|  |         } | ||
|  | 
 | ||
|  |         public IAsyncResult BeginReceive(TimeSpan timeout, AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             if (timeout < TimeSpan.Zero) | ||
|  |             { | ||
|  |                 throw FxTrace.Exception.AsError(new ArgumentOutOfRangeException("timeout", timeout, SR.TimeoutOutOfRange0)); | ||
|  |             } | ||
|  | 
 | ||
|  |             this.ThrowPending(); | ||
|  |             return InputChannel.HelpBeginReceive(this, timeout, callback, state); | ||
|  |         } | ||
|  | 
 | ||
|  |         public Message EndReceive(IAsyncResult result) | ||
|  |         { | ||
|  |             return InputChannel.HelpEndReceive(result); | ||
|  |         } | ||
|  | 
 | ||
|  |         public bool TryReceive(TimeSpan timeout, out Message message) | ||
|  |         { | ||
|  |             if (timeout < TimeSpan.Zero) | ||
|  |             { | ||
|  |                 throw FxTrace.Exception.AsError(new ArgumentOutOfRangeException("timeout", timeout, SR.TimeoutOutOfRange0)); | ||
|  |             } | ||
|  | 
 | ||
|  |             this.ThrowPending(); | ||
|  |             return base.Dequeue(timeout, out message); | ||
|  |         } | ||
|  | 
 | ||
|  |         public IAsyncResult BeginTryReceive(TimeSpan timeout, AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             if (timeout < TimeSpan.Zero) | ||
|  |             { | ||
|  |                 throw FxTrace.Exception.AsError(new ArgumentOutOfRangeException("timeout", timeout, SR.TimeoutOutOfRange0)); | ||
|  |             } | ||
|  | 
 | ||
|  |             this.ThrowPending(); | ||
|  |             return base.BeginDequeue(timeout, callback, state); | ||
|  |         } | ||
|  | 
 | ||
|  |         public bool EndTryReceive(IAsyncResult result, out Message message) | ||
|  |         { | ||
|  |             return base.EndDequeue(result, out message); | ||
|  |         } | ||
|  | 
 | ||
|  |         public bool WaitForMessage(TimeSpan timeout) | ||
|  |         { | ||
|  |             if (timeout < TimeSpan.Zero) | ||
|  |             { | ||
|  |                 throw FxTrace.Exception.AsError(new ArgumentOutOfRangeException("timeout", timeout, SR.TimeoutOutOfRange0)); | ||
|  |             } | ||
|  | 
 | ||
|  |             this.ThrowPending(); | ||
|  |             return base.WaitForItem(timeout); | ||
|  |         } | ||
|  | 
 | ||
|  |         public IAsyncResult BeginWaitForMessage(TimeSpan timeout, AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             if (timeout < TimeSpan.Zero) | ||
|  |             { | ||
|  |                 throw FxTrace.Exception.AsError(new ArgumentOutOfRangeException("timeout", timeout, SR.TimeoutOutOfRange0)); | ||
|  |             } | ||
|  | 
 | ||
|  |             this.ThrowPending(); | ||
|  |             return base.BeginWaitForItem(timeout, callback, state); | ||
|  |         } | ||
|  | 
 | ||
|  |         public bool EndWaitForMessage(IAsyncResult result) | ||
|  |         { | ||
|  |             return base.EndWaitForItem(result); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal override void FinishEnqueueMessage(Message message, Action dequeuedCallback, bool canDispatchOnThisThread) | ||
|  |         { | ||
|  |             if (!this.IsMulticast) | ||
|  |             { | ||
|  |                 //When using Multicast, we can't assume that receiving one message means that we are done receiving messages. | ||
|  |                 //For example, Discovery will send one message out and receive n responses that match.  Because of this, we  | ||
|  |                 //can only short circuit retransmission when using unicast. | ||
|  |                 this.UdpOutputChannel.CancelRetransmission(message.Headers.RelatesTo); | ||
|  |             } | ||
|  |             this.EnqueueAndDispatch(message, dequeuedCallback, canDispatchOnThisThread); | ||
|  |         } | ||
|  |     } | ||
|  | } |