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);
 | |
|         }
 | |
|     }
 | |
| }
 |