You've already forked linux-packaging-mono
Imported Upstream version 4.6.0.125
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
This commit is contained in:
parent
a569aebcfd
commit
e79aa3c0ed
@ -0,0 +1,222 @@
|
||||
//------------------------------------------------------------
|
||||
// Copyright (c) Microsoft Corporation. All rights reserved.
|
||||
//------------------------------------------------------------
|
||||
namespace System.ServiceModel.Channels
|
||||
{
|
||||
using System.Diagnostics;
|
||||
using System.Runtime;
|
||||
using System.ServiceModel;
|
||||
using System.Threading;
|
||||
|
||||
delegate void ConnectionModeCallback(ConnectionModeReader connectionModeReader);
|
||||
|
||||
sealed class ConnectionModeReader : InitialServerConnectionReader
|
||||
{
|
||||
Exception readException;
|
||||
ServerModeDecoder decoder;
|
||||
byte[] buffer;
|
||||
int offset;
|
||||
int size;
|
||||
ConnectionModeCallback callback;
|
||||
static WaitCallback readCallback;
|
||||
TimeoutHelper receiveTimeoutHelper;
|
||||
|
||||
public ConnectionModeReader(IConnection connection, ConnectionModeCallback callback, ConnectionClosedCallback closedCallback)
|
||||
: base(connection, closedCallback)
|
||||
{
|
||||
this.callback = callback;
|
||||
}
|
||||
|
||||
public int BufferOffset
|
||||
{
|
||||
get { return offset; }
|
||||
}
|
||||
|
||||
public int BufferSize
|
||||
{
|
||||
get { return size; }
|
||||
}
|
||||
|
||||
public long StreamPosition
|
||||
{
|
||||
get { return decoder.StreamPosition; }
|
||||
}
|
||||
|
||||
public TimeSpan GetRemainingTimeout()
|
||||
{
|
||||
return this.receiveTimeoutHelper.RemainingTime();
|
||||
}
|
||||
|
||||
void Complete(Exception e)
|
||||
{
|
||||
// exception will be logged by the caller
|
||||
readException = e;
|
||||
Complete();
|
||||
}
|
||||
|
||||
void Complete()
|
||||
{
|
||||
callback(this);
|
||||
}
|
||||
|
||||
bool ContinueReading()
|
||||
{
|
||||
for (;;)
|
||||
{
|
||||
if (size == 0)
|
||||
{
|
||||
if (readCallback == null)
|
||||
{
|
||||
readCallback = new WaitCallback(ReadCallback);
|
||||
}
|
||||
|
||||
if (Connection.BeginRead(0, Connection.AsyncReadBufferSize, GetRemainingTimeout(),
|
||||
readCallback, this) == AsyncCompletionResult.Queued)
|
||||
{
|
||||
break;
|
||||
}
|
||||
if (!GetReadResult()) // we're at EOF, bail
|
||||
{
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
for (;;)
|
||||
{
|
||||
int bytesDecoded;
|
||||
try
|
||||
{
|
||||
bytesDecoded = decoder.Decode(buffer, offset, size);
|
||||
}
|
||||
catch (CommunicationException e)
|
||||
{
|
||||
// see if we need to send back a framing fault
|
||||
string framingFault;
|
||||
if (FramingEncodingString.TryGetFaultString(e, out framingFault))
|
||||
{
|
||||
byte[] drainBuffer = new byte[128];
|
||||
InitialServerConnectionReader.SendFault(
|
||||
Connection, framingFault, drainBuffer, GetRemainingTimeout(),
|
||||
MaxViaSize + MaxContentTypeSize);
|
||||
base.Close(GetRemainingTimeout());
|
||||
}
|
||||
throw;
|
||||
}
|
||||
|
||||
if (bytesDecoded > 0)
|
||||
{
|
||||
offset += bytesDecoded;
|
||||
size -= bytesDecoded;
|
||||
}
|
||||
if (decoder.CurrentState == ServerModeDecoder.State.Done)
|
||||
{
|
||||
return true;
|
||||
}
|
||||
if (size == 0)
|
||||
{
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static void ReadCallback(object state)
|
||||
{
|
||||
ConnectionModeReader reader = (ConnectionModeReader)state;
|
||||
bool completeSelf = false;
|
||||
Exception completionException = null;
|
||||
try
|
||||
{
|
||||
if (reader.GetReadResult())
|
||||
{
|
||||
completeSelf = reader.ContinueReading();
|
||||
}
|
||||
}
|
||||
#pragma warning suppress 56500 // [....], transferring exception to caller
|
||||
catch (Exception e)
|
||||
{
|
||||
if (Fx.IsFatal(e))
|
||||
{
|
||||
throw;
|
||||
}
|
||||
|
||||
completeSelf = true;
|
||||
completionException = e;
|
||||
}
|
||||
|
||||
if (completeSelf)
|
||||
{
|
||||
reader.Complete(completionException);
|
||||
}
|
||||
}
|
||||
|
||||
bool GetReadResult()
|
||||
{
|
||||
offset = 0;
|
||||
size = Connection.EndRead();
|
||||
if (size == 0)
|
||||
{
|
||||
if (this.decoder.StreamPosition == 0) // client timed out a cached connection
|
||||
{
|
||||
base.Close(GetRemainingTimeout());
|
||||
return false;
|
||||
}
|
||||
else
|
||||
{
|
||||
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(decoder.CreatePrematureEOFException());
|
||||
}
|
||||
}
|
||||
|
||||
// restore ExceptionEventType to Error after the initial read for cached connections
|
||||
Connection.ExceptionEventType = TraceEventType.Error;
|
||||
|
||||
if (buffer == null)
|
||||
{
|
||||
buffer = Connection.AsyncReadBuffer;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
public FramingMode GetConnectionMode()
|
||||
{
|
||||
if (readException != null)
|
||||
{
|
||||
throw DiagnosticUtility.ExceptionUtility.ThrowHelper(readException, Connection.ExceptionEventType);
|
||||
}
|
||||
|
||||
return decoder.Mode;
|
||||
}
|
||||
|
||||
public void StartReading(TimeSpan receiveTimeout, Action connectionDequeuedCallback)
|
||||
{
|
||||
this.decoder = new ServerModeDecoder();
|
||||
this.receiveTimeoutHelper = new TimeoutHelper(receiveTimeout);
|
||||
this.ConnectionDequeuedCallback = connectionDequeuedCallback;
|
||||
bool completeSelf = false;
|
||||
Exception completionException = null;
|
||||
try
|
||||
{
|
||||
completeSelf = ContinueReading();
|
||||
}
|
||||
#pragma warning suppress 56500 // [....], transferring exception to caller
|
||||
catch (Exception e)
|
||||
{
|
||||
if (Fx.IsFatal(e))
|
||||
{
|
||||
throw;
|
||||
}
|
||||
|
||||
completeSelf = true;
|
||||
completionException = e;
|
||||
}
|
||||
|
||||
if (completeSelf)
|
||||
{
|
||||
Complete(completionException);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user