1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
|
//------------------------------------------------------------
// Copyright (c) Microsoft Corporation. All rights reserved.
//------------------------------------------------------------
namespace System.ServiceModel.Channels
{
using System.IO;
using System.Xml;
using System.Diagnostics;
abstract class BufferedMessageWriter
{
int[] sizeHistory;
int sizeHistoryIndex;
const int sizeHistoryCount = 4;
const int expectedSizeVariance = 256;
BufferManagerOutputStream stream;
public BufferedMessageWriter()
{
this.stream = new BufferManagerOutputStream(SR.MaxSentMessageSizeExceeded);
InitMessagePredicter();
}
protected abstract XmlDictionaryWriter TakeXmlWriter(Stream stream);
protected abstract void ReturnXmlWriter(XmlDictionaryWriter writer);
public ArraySegment<byte> WriteMessage(Message message, BufferManager bufferManager, int initialOffset, int maxSizeQuota)
{
int effectiveMaxSize;
// make sure that maxSize has room for initialOffset without overflowing, since
// the effective buffer size is message size + initialOffset
if (maxSizeQuota <= int.MaxValue - initialOffset)
effectiveMaxSize = maxSizeQuota + initialOffset;
else
effectiveMaxSize = int.MaxValue;
int predictedMessageSize = PredictMessageSize();
if (predictedMessageSize > effectiveMaxSize)
predictedMessageSize = effectiveMaxSize;
else if (predictedMessageSize < initialOffset)
predictedMessageSize = initialOffset;
try
{
stream.Init(predictedMessageSize, maxSizeQuota, effectiveMaxSize, bufferManager);
stream.Skip(initialOffset);
XmlDictionaryWriter writer = TakeXmlWriter(stream);
OnWriteStartMessage(writer);
message.WriteMessage(writer);
OnWriteEndMessage(writer);
writer.Flush();
ReturnXmlWriter(writer);
int size;
byte[] buffer = stream.ToArray(out size);
RecordActualMessageSize(size);
return new ArraySegment<byte>(buffer, initialOffset, size - initialOffset);
}
finally
{
stream.Clear();
}
}
protected virtual void OnWriteStartMessage(XmlDictionaryWriter writer)
{
}
protected virtual void OnWriteEndMessage(XmlDictionaryWriter writer)
{
}
void InitMessagePredicter()
{
sizeHistory = new int[4];
for (int i = 0; i < sizeHistoryCount; i++)
sizeHistory[i] = 256;
}
int PredictMessageSize()
{
int max = 0;
for (int i = 0; i < sizeHistoryCount; i++)
if (sizeHistory[i] > max)
max = sizeHistory[i];
return max + expectedSizeVariance;
}
void RecordActualMessageSize(int size)
{
sizeHistory[sizeHistoryIndex] = size;
sizeHistoryIndex = (sizeHistoryIndex + 1) % sizeHistoryCount;
}
}
}
|