e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
765 lines
21 KiB
C#
765 lines
21 KiB
C#
//------------------------------------------------------------
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//------------------------------------------------------------
|
|
namespace System.ServiceModel.Dispatcher
|
|
{
|
|
using System;
|
|
using System.Runtime;
|
|
|
|
internal struct StackFrame
|
|
{
|
|
internal int basePtr;
|
|
internal int endPtr;
|
|
|
|
#if NO
|
|
internal StackFrame(int basePtr)
|
|
{
|
|
Fx.Assert(basePtr >= 0, "");
|
|
this.basePtr = basePtr;
|
|
this.endPtr = this.basePtr - 1;
|
|
}
|
|
|
|
internal StackFrame(int basePtr, int count)
|
|
{
|
|
Fx.Assert(basePtr >= 0, "");
|
|
this.basePtr = basePtr;
|
|
this.endPtr = basePtr + count - 1;
|
|
}
|
|
#endif
|
|
internal int Count
|
|
{
|
|
get
|
|
{
|
|
return this.endPtr - this.basePtr + 1;
|
|
}
|
|
}
|
|
|
|
internal int EndPtr
|
|
{
|
|
#if NO
|
|
get
|
|
{
|
|
return this.endPtr;
|
|
}
|
|
#endif
|
|
set
|
|
{
|
|
Fx.Assert(value >= this.basePtr, "");
|
|
this.endPtr = value;
|
|
}
|
|
}
|
|
#if NO
|
|
internal void Clear()
|
|
{
|
|
this.endPtr = this.basePtr - 1;
|
|
}
|
|
#endif
|
|
internal int this[int offset]
|
|
{
|
|
get
|
|
{
|
|
Fx.Assert(this.IsValidPtr(this.basePtr + offset), "");
|
|
return this.basePtr + offset;
|
|
}
|
|
}
|
|
#if NO
|
|
internal void Set(int basePtr)
|
|
{
|
|
Fx.Assert(basePtr >= 0, "");
|
|
this.basePtr = basePtr;
|
|
this.endPtr = this.basePtr - 1;
|
|
}
|
|
#endif
|
|
internal bool IsValidPtr(int ptr)
|
|
{
|
|
return (ptr >= this.basePtr && ptr <= this.endPtr);
|
|
}
|
|
}
|
|
|
|
internal struct StackRegion
|
|
{
|
|
internal QueryRange bounds;
|
|
internal int stackPtr;
|
|
|
|
internal StackRegion(QueryRange bounds)
|
|
{
|
|
this.bounds = bounds;
|
|
this.stackPtr = bounds.start - 1;
|
|
}
|
|
#if NO
|
|
internal StackRegion(QueryRange bounds, int stackPtr)
|
|
{
|
|
Fx.Assert(bounds.IsInRange(stackPtr), "");
|
|
this.bounds = bounds;
|
|
this.stackPtr = stackPtr;
|
|
}
|
|
#endif
|
|
internal int Count
|
|
{
|
|
get
|
|
{
|
|
return this.stackPtr - this.bounds.start + 1;
|
|
}
|
|
}
|
|
#if NO
|
|
internal bool IsReady
|
|
{
|
|
get
|
|
{
|
|
return this.bounds.IsNotEmpty;
|
|
}
|
|
}
|
|
#endif
|
|
internal bool NeedsGrowth
|
|
{
|
|
get
|
|
{
|
|
return (this.stackPtr > this.bounds.end);
|
|
}
|
|
}
|
|
|
|
internal void Clear()
|
|
{
|
|
this.stackPtr = this.bounds.start - 1;
|
|
}
|
|
|
|
internal void Grow(int growBy)
|
|
{
|
|
this.bounds.end += growBy;
|
|
}
|
|
|
|
internal bool IsValidStackPtr()
|
|
{
|
|
return this.bounds.IsInRange(this.stackPtr);
|
|
}
|
|
|
|
internal bool IsValidStackPtr(int stackPtr)
|
|
{
|
|
return this.bounds.IsInRange(stackPtr);
|
|
}
|
|
#if NO
|
|
internal void Set(int start, int end)
|
|
{
|
|
this.bounds.Set(start, end);
|
|
this.stackPtr += start;
|
|
}
|
|
#endif
|
|
internal void Shift(int shiftBy)
|
|
{
|
|
this.bounds.Shift(shiftBy);
|
|
this.stackPtr += shiftBy;
|
|
}
|
|
}
|
|
|
|
// The eval stack as well as all its contained data structures are STRUCTs
|
|
// fast to allocate
|
|
internal struct EvalStack
|
|
{
|
|
internal QueryBuffer<Value> buffer;
|
|
internal StackRegion frames;
|
|
internal StackRegion stack;
|
|
internal const int DefaultSize = 2;
|
|
internal bool contextOnTopOfStack;
|
|
|
|
internal EvalStack(int frameCapacity, int stackCapacity)
|
|
{
|
|
Fx.Assert(frameCapacity >= 0 && stackCapacity >= 0, "");
|
|
|
|
// All structs! Cost of allocation is relatively mild...
|
|
this.buffer = new QueryBuffer<Value>(frameCapacity + stackCapacity);
|
|
this.stack = new StackRegion(new QueryRange(0, stackCapacity - 1));
|
|
this.buffer.Reserve(stackCapacity);
|
|
this.frames = new StackRegion(new QueryRange(stackCapacity, stackCapacity + frameCapacity - 1));
|
|
this.buffer.Reserve(frameCapacity);
|
|
this.contextOnTopOfStack = false;
|
|
}
|
|
#if NO
|
|
internal EvalStack(ref EvalStack stack)
|
|
{
|
|
this.buffer = new QueryBuffer<Value>(stack.buffer);
|
|
this.stackCapacity = stack.stackCapacity;
|
|
this.frameCapacity = stack.frameCapacity;
|
|
this.stack = stack.stack;
|
|
this.frames = stack.frames;
|
|
}
|
|
#endif
|
|
internal Value[] Buffer
|
|
{
|
|
get
|
|
{
|
|
return this.buffer.buffer;
|
|
}
|
|
}
|
|
#if NO
|
|
internal int FrameCount
|
|
{
|
|
get
|
|
{
|
|
return this.frames.Count;
|
|
}
|
|
}
|
|
|
|
internal int FramePtr
|
|
{
|
|
get
|
|
{
|
|
return this.frames.stackPtr;
|
|
}
|
|
}
|
|
#endif
|
|
internal StackFrame this[int frameIndex]
|
|
{
|
|
get
|
|
{
|
|
return this.buffer.buffer[this.frames.stackPtr - frameIndex].Frame;
|
|
}
|
|
}
|
|
#if NO
|
|
internal bool IsReady
|
|
{
|
|
get
|
|
{
|
|
return (this.buffer.count > 0);
|
|
}
|
|
}
|
|
#endif
|
|
internal StackFrame SecondArg
|
|
{
|
|
get
|
|
{
|
|
return this[1];
|
|
}
|
|
}
|
|
#if NO
|
|
internal int StackPtr
|
|
{
|
|
get
|
|
{
|
|
return this.stack.stackPtr;
|
|
}
|
|
}
|
|
#endif
|
|
internal StackFrame TopArg
|
|
{
|
|
get
|
|
{
|
|
return this[0];
|
|
}
|
|
}
|
|
|
|
internal void Clear()
|
|
{
|
|
this.stack.Clear();
|
|
this.frames.Clear();
|
|
this.contextOnTopOfStack = false;
|
|
}
|
|
|
|
internal void CopyFrom(ref EvalStack stack)
|
|
{
|
|
this.buffer.CopyFrom(ref stack.buffer);
|
|
this.frames = stack.frames;
|
|
this.stack = stack.stack;
|
|
this.contextOnTopOfStack = stack.contextOnTopOfStack;
|
|
}
|
|
|
|
internal int CalculateNodecount()
|
|
{
|
|
if (this.stack.stackPtr < 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
StackFrame topFrame = this.TopArg;
|
|
int count = 0;
|
|
|
|
for (int i = topFrame.basePtr; i <= topFrame.endPtr; ++i)
|
|
{
|
|
Fx.Assert(this.buffer[i].IsType(ValueDataType.Sequence), "");
|
|
count += this.buffer[i].NodeCount;
|
|
}
|
|
|
|
return count;
|
|
}
|
|
#if NO
|
|
internal void Erase()
|
|
{
|
|
this.buffer.Erase();
|
|
}
|
|
#endif
|
|
void GrowFrames()
|
|
{
|
|
int growBy = this.frames.Count;
|
|
this.buffer.ReserveAt(this.frames.bounds.end + 1, growBy);
|
|
this.frames.Grow(growBy);
|
|
}
|
|
|
|
void GrowStack(int growthNeeded)
|
|
{
|
|
int growBy = this.stack.bounds.Count;
|
|
if (growthNeeded > growBy)
|
|
{
|
|
growBy = growthNeeded;
|
|
}
|
|
|
|
this.buffer.ReserveAt(this.stack.bounds.end + 1, growBy);
|
|
this.stack.Grow(growBy);
|
|
this.frames.Shift(growBy);
|
|
}
|
|
#if NO
|
|
internal void Init()
|
|
{
|
|
this.buffer.Reserve(this.stackCapacity);
|
|
this.stack.Set(0, stackCapacity - 1);
|
|
this.buffer.Reserve(this.frameCapacity);
|
|
this.frames.Set(stackCapacity, stackCapacity + frameCapacity - 1);
|
|
}
|
|
|
|
internal void Init(Value[] buffer, int stackCapacity, int frameCapacity)
|
|
{
|
|
Fx.Assert(null != buffer, "");
|
|
this.stackCapacity = stackCapacity;
|
|
this.frameCapacity = frameCapacity;
|
|
|
|
this.buffer = new QueryBuffer<Value>(buffer);
|
|
this.stack = new StackRegion(new QueryRange(0, stackCapacity - 1));
|
|
this.buffer.Reserve(stackCapacity);
|
|
this.frames = new StackRegion(new QueryRange(stackCapacity, stackCapacity + frameCapacity - 1));
|
|
this.buffer.Reserve(frameCapacity);
|
|
}
|
|
#endif
|
|
internal bool InUse
|
|
{
|
|
get
|
|
{
|
|
if (contextOnTopOfStack)
|
|
return (this.frames.Count > 1);
|
|
else
|
|
return (this.frames.Count > 0);
|
|
}
|
|
}
|
|
|
|
internal bool PeekBoolean(int index)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
return this.buffer.buffer[index].GetBoolean();
|
|
}
|
|
|
|
internal double PeekDouble(int index)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
return this.buffer.buffer[index].GetDouble();
|
|
}
|
|
#if NO
|
|
internal int PeekInteger(int index)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
return (int)this.buffer.buffer[index].GetDouble();
|
|
}
|
|
#endif
|
|
internal NodeSequence PeekSequence(int index)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
return this.buffer.buffer[index].GetSequence();
|
|
}
|
|
|
|
internal string PeekString(int index)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
return this.buffer.buffer[index].GetString();
|
|
}
|
|
#if NO
|
|
internal void Pop()
|
|
{
|
|
this.stack.stackPtr--;
|
|
Fx.Assert(this.stack.IsValidStackPtr(), "");
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void PopFrame()
|
|
{
|
|
Fx.Assert(this.frames.IsValidStackPtr(), "");
|
|
this.stack.stackPtr = this.buffer.buffer[this.frames.stackPtr].StackPtr;
|
|
this.frames.stackPtr--;
|
|
}
|
|
#endif
|
|
internal void PopFrame(ProcessingContext context)
|
|
{
|
|
Fx.Assert(this.frames.IsValidStackPtr(), "");
|
|
|
|
StackFrame topArg = this.TopArg;
|
|
for (int i = topArg.basePtr; i <= topArg.endPtr; ++i)
|
|
{
|
|
this.buffer.buffer[i].Clear(context);
|
|
}
|
|
|
|
this.stack.stackPtr = topArg.basePtr - 1;
|
|
this.frames.stackPtr--;
|
|
}
|
|
|
|
internal void PushFrame()
|
|
{
|
|
this.frames.stackPtr++;
|
|
if (this.frames.NeedsGrowth)
|
|
{
|
|
this.GrowFrames();
|
|
}
|
|
|
|
//
|
|
// The first element in the new frame will be the NEXT item pushed onto the stack
|
|
// We save offsets because stacks may get moved and repositioned
|
|
//
|
|
this.buffer.buffer[this.frames.stackPtr].StartFrame(this.stack.stackPtr);
|
|
}
|
|
|
|
internal void PopSequenceFrameTo(ref EvalStack dest)
|
|
{
|
|
StackFrame topFrame = this.TopArg;
|
|
|
|
dest.PushFrame();
|
|
int argCount = topFrame.Count;
|
|
switch (argCount)
|
|
{
|
|
default:
|
|
dest.Push(this.buffer.buffer, topFrame.basePtr, argCount);
|
|
break;
|
|
|
|
case 0:
|
|
break;
|
|
|
|
case 1:
|
|
dest.Push(this.buffer.buffer[topFrame.basePtr].Sequence);
|
|
break;
|
|
}
|
|
|
|
// Pop original fame
|
|
this.stack.stackPtr = topFrame.basePtr - 1;
|
|
this.frames.stackPtr--;
|
|
}
|
|
#if NO
|
|
internal void Push()
|
|
{
|
|
this.stack.stackPtr++;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(1);
|
|
}
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(int count)
|
|
{
|
|
this.stack.stackPtr += count;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(count);
|
|
}
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
#endif
|
|
|
|
internal void Push(string val)
|
|
{
|
|
this.stack.stackPtr++;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(1);
|
|
}
|
|
|
|
this.buffer.buffer[this.stack.stackPtr].String = val;
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(string val, int addCount)
|
|
{
|
|
int stackPtr = this.stack.stackPtr;
|
|
this.stack.stackPtr += addCount;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(addCount);
|
|
}
|
|
|
|
int stackMax = stackPtr + addCount;
|
|
while (stackPtr < stackMax)
|
|
{
|
|
this.buffer.buffer[++stackPtr].String = val;
|
|
}
|
|
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(bool val)
|
|
{
|
|
this.stack.stackPtr++;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(1);
|
|
}
|
|
|
|
this.buffer.buffer[this.stack.stackPtr].Boolean = val;
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(bool val, int addCount)
|
|
{
|
|
int stackPtr = this.stack.stackPtr;
|
|
|
|
this.stack.stackPtr += addCount;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(addCount);
|
|
}
|
|
|
|
int stackMax = stackPtr + addCount;
|
|
|
|
while (stackPtr < stackMax)
|
|
{
|
|
this.buffer.buffer[++stackPtr].Boolean = val;
|
|
}
|
|
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(double val)
|
|
{
|
|
this.stack.stackPtr++;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(1);
|
|
}
|
|
|
|
this.buffer.buffer[this.stack.stackPtr].Double = val;
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(double val, int addCount)
|
|
{
|
|
int stackPtr = this.stack.stackPtr;
|
|
|
|
this.stack.stackPtr += addCount;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(addCount);
|
|
}
|
|
|
|
int stackMax = stackPtr + addCount;
|
|
|
|
while (stackPtr < stackMax)
|
|
{
|
|
this.buffer.buffer[++stackPtr].Double = val;
|
|
}
|
|
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(NodeSequence val)
|
|
{
|
|
this.stack.stackPtr++;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(1);
|
|
}
|
|
|
|
this.buffer.buffer[this.stack.stackPtr].Sequence = val;
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(NodeSequence val, int addCount)
|
|
{
|
|
// One of the addCount refs was added by the call to CreateSequence
|
|
val.refCount += addCount - 1;
|
|
|
|
int stackPtr = this.stack.stackPtr;
|
|
|
|
this.stack.stackPtr += addCount;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(addCount);
|
|
}
|
|
|
|
int stackMax = stackPtr + addCount;
|
|
|
|
while (stackPtr < stackMax)
|
|
{
|
|
this.buffer.buffer[++stackPtr].Sequence = val;
|
|
}
|
|
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
|
|
internal void Push(Value[] buffer, int startAt, int addCount)
|
|
{
|
|
if (addCount > 0)
|
|
{
|
|
int stackPtr = this.stack.stackPtr + 1;
|
|
this.stack.stackPtr += addCount;
|
|
if (this.stack.NeedsGrowth)
|
|
{
|
|
this.GrowStack(addCount);
|
|
}
|
|
|
|
if (1 == addCount)
|
|
{
|
|
this.buffer.buffer[stackPtr] = buffer[startAt];
|
|
}
|
|
else
|
|
{
|
|
Array.Copy(buffer, startAt, this.buffer.buffer, stackPtr, addCount);
|
|
}
|
|
this.buffer.buffer[this.frames.stackPtr].FrameEndPtr = this.stack.stackPtr;
|
|
}
|
|
}
|
|
#if NO
|
|
internal void Push(ref EvalStack source)
|
|
{
|
|
this.Push(source.buffer.buffer, source.stack.bounds.start, source.frames.bounds.end + 1);
|
|
}
|
|
#endif
|
|
internal void ReplaceAt(int index, NodeSequence seq)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index) && this.buffer.buffer[index].IsType(ValueDataType.Sequence), "");
|
|
this.buffer.buffer[index].Sequence = seq;
|
|
}
|
|
|
|
internal void SetValue(ProcessingContext context, int index, bool val)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
this.buffer.buffer[index].Update(context, val);
|
|
}
|
|
|
|
internal void SetValue(ProcessingContext context, int index, double val)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
this.buffer.buffer[index].Update(context, val);
|
|
}
|
|
|
|
internal void SetValue(ProcessingContext context, int index, string val)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
this.buffer.buffer[index].Update(context, val);
|
|
}
|
|
|
|
internal void SetValue(ProcessingContext context, int index, NodeSequence val)
|
|
{
|
|
Fx.Assert(this.stack.IsValidStackPtr(index), "");
|
|
this.buffer.buffer[index].Update(context, val);
|
|
}
|
|
|
|
internal void TransferPositionsTo(ref EvalStack stack)
|
|
{
|
|
StackFrame arg = this.TopArg;
|
|
|
|
stack.PushFrame();
|
|
for (int i = arg.basePtr; i <= arg.endPtr; ++i)
|
|
{
|
|
NodeSequence seq = this.buffer.buffer[i].Sequence;
|
|
int nodeCount = seq.Count;
|
|
|
|
if ((this.stack.stackPtr + nodeCount) > this.stack.bounds.end)
|
|
{
|
|
this.GrowStack(nodeCount);
|
|
}
|
|
|
|
for (int n = 0; n < nodeCount; ++n)
|
|
{
|
|
stack.Push((double)seq.Items[n].Position);
|
|
}
|
|
}
|
|
}
|
|
|
|
internal void TransferSequenceSizeTo(ref EvalStack stack)
|
|
{
|
|
StackFrame arg = this.TopArg;
|
|
|
|
stack.PushFrame();
|
|
for (int i = arg.basePtr; i <= arg.endPtr; ++i)
|
|
{
|
|
NodeSequence seq = this.buffer.buffer[i].Sequence;
|
|
int nodeCount = seq.Count;
|
|
|
|
if ((this.stack.stackPtr + nodeCount) > this.stack.bounds.end)
|
|
{
|
|
this.GrowStack(nodeCount);
|
|
}
|
|
|
|
for (int n = 0; n < nodeCount; ++n)
|
|
{
|
|
stack.Push((double)NodeSequence.GetContextSize(seq, n));
|
|
}
|
|
}
|
|
}
|
|
#if NO
|
|
internal void Trim()
|
|
{
|
|
this.buffer.TrimToCount();
|
|
}
|
|
#endif
|
|
}
|
|
#if NO
|
|
internal struct BoundedStack<T>
|
|
{
|
|
QueryBuffer<T> buffer;
|
|
int maxSize;
|
|
|
|
internal BoundedStack(int capacity)
|
|
{
|
|
this.buffer = new QueryBuffer<T>(0);
|
|
this.maxSize = capacity;
|
|
}
|
|
|
|
internal bool HasItems
|
|
{
|
|
get
|
|
{
|
|
return (this.buffer.count > 0);
|
|
}
|
|
}
|
|
|
|
internal bool HasSpace
|
|
{
|
|
get
|
|
{
|
|
return (this.buffer.count < this.maxSize);
|
|
}
|
|
}
|
|
|
|
internal int MaxSize
|
|
{
|
|
get
|
|
{
|
|
return this.maxSize;
|
|
}
|
|
set
|
|
{
|
|
Fx.Assert(value >= 0, "");
|
|
this.maxSize = value;
|
|
if (value < this.buffer.count)
|
|
{
|
|
this.buffer.count = value;
|
|
this.buffer.TrimToCount();
|
|
}
|
|
}
|
|
}
|
|
|
|
internal T Pop()
|
|
{
|
|
return this.buffer.Pop();
|
|
}
|
|
|
|
internal void Push(T t)
|
|
{
|
|
if (this.buffer.count == this.maxSize)
|
|
{
|
|
return;
|
|
}
|
|
|
|
this.buffer.Push(t);
|
|
}
|
|
|
|
internal void Trim()
|
|
{
|
|
this.buffer.TrimToCount();
|
|
}
|
|
}
|
|
#endif
|
|
}
|