e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
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);
|
|
}
|
|
}
|
|
}
|