| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470 |
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Text;
- using ILRuntime.CLR.Method;
- using ILRuntime.CLR.TypeSystem;
- using ILRuntime.Other;
- using ILRuntime.Runtime.Intepreter;
- namespace ILRuntime.Runtime.Stack
- {
- unsafe class RuntimeStack : IDisposable
- {
- ILIntepreter intepreter;
- StackObject* pointer;
- StackObject* endOfMemory;
- StackObject* valueTypePtr;
- IntPtr nativePointer;
- #if DEBUG && !DISABLE_ILRUNTIME_DEBUG
- IList<object> managedStack = new List<object>(32);
- #else
- IList<object> managedStack = new UncheckedList<object>(32);
- #endif
- UncheckedStack<StackFrame> frames = new UncheckedStack<StackFrame>();
- public const int MAXIMAL_STACK_OBJECTS = 1024 * 16;
- public UncheckedStack<StackFrame> Frames { get { return frames; } }
- public RuntimeStack(ILIntepreter intepreter)
- {
- this.intepreter = intepreter;
- nativePointer = System.Runtime.InteropServices.Marshal.AllocHGlobal(sizeof(StackObject) * MAXIMAL_STACK_OBJECTS);
- pointer = (StackObject*)nativePointer.ToPointer();
- endOfMemory = Add(pointer, MAXIMAL_STACK_OBJECTS);
- valueTypePtr = endOfMemory - 1;
- }
- ~RuntimeStack()
- {
- Dispose();
- }
- public StackObject* StackBase
- {
- get
- {
- return pointer;
- }
- }
- public StackObject* ValueTypeStackPointer
- {
- get
- {
- return valueTypePtr;
- }
- private set
- {
- if (value > ValueTypeStackBase)
- throw new StackOverflowException();
- valueTypePtr = value;
- }
- }
- public StackObject* ValueTypeStackBase
- {
- get
- {
- return endOfMemory - 1;
- }
- }
- public IList<object> ManagedStack { get { return managedStack; } }
- public void ResetValueTypePointer()
- {
- valueTypePtr = endOfMemory - 1;
- }
- public void InitializeFrame(ILMethod method, StackObject* esp, out StackFrame res)
- {
- #if DEBUG
- if (esp < pointer || esp >= endOfMemory)
- throw new StackOverflowException();
- if (frames.Count > 0 && frames.Peek().BasePointer > esp)
- throw new StackOverflowException();
- #endif
- res = new StackFrame();
- res.LocalVarPointer = esp;
- res.Method = method;
- #if DEBUG && !DISABLE_ILRUNTIME_DEBUG
- res.Address = new IntegerReference();
- for (int i = 0; i < method.LocalVariableCount; i++)
- {
- var p = Add(esp, i);
- p->ObjectType = ObjectTypes.Null;
- }
- #endif
- res.BasePointer = method.LocalVariableCount > 0 ? Add(esp, method.LocalVariableCount) : esp;
- res.ManagedStackBase = managedStack.Count;
- res.ValueTypeBasePointer = valueTypePtr;
- //frames.Push(res);
- }
- public void PushFrame(ref StackFrame frame)
- {
- frames.Push(ref frame);
- }
- public StackObject* PopFrame(ref StackFrame frame, StackObject* esp)
- {
- #if DEBUG
- if (frames.Count > 0 && frames.Peek().BasePointer == frame.BasePointer)
- #endif
- frames.Pop();
- #if DEBUG
- else
- throw new NotSupportedException();
- #endif
- StackObject* returnVal = esp - 1;
- var method = frame.Method;
- StackObject* ret = ILIntepreter.Minus(frame.LocalVarPointer, method.ParameterCount);
- int mStackBase = frame.ManagedStackBase;
- if (method.HasThis)
- ret--;
- for (StackObject* ptr = ret; ptr < frame.LocalVarPointer; ptr++)
- {
- if (ptr->ObjectType == ObjectTypes.ValueTypeObjectReference)
- {
- var addr = ILIntepreter.ResolveReference(ptr);
- int start = int.MaxValue;
- int end = int.MaxValue;
- var tmp = addr;
- CountValueTypeManaged(ptr, ref start, ref end, &tmp);
- if (addr > frame.ValueTypeBasePointer)
- {
- frame.ValueTypeBasePointer = addr;
- }
- if (start < mStackBase)
- mStackBase = start;
- }
- }
- if(method.ReturnType != intepreter.AppDomain.VoidType)
- {
- *ret = *returnVal;
- if(ret->ObjectType == ObjectTypes.Object)
- {
- ret->Value = mStackBase;
- managedStack[mStackBase] = managedStack[returnVal->Value];
- mStackBase++;
- }
- else if(ret->ObjectType == ObjectTypes.ValueTypeObjectReference)
- {
- StackObject* oriAddr = frame.ValueTypeBasePointer;
- RelocateValueType(ret, ref frame.ValueTypeBasePointer, ref mStackBase);
- *(long*)&ret->Value = (long)oriAddr;
- }
- ret++;
- }
- #if DEBUG && !DISABLE_ILRUNTIME_DEBUG
- ((List<object>)managedStack).RemoveRange(mStackBase, managedStack.Count - mStackBase);
- #else
- ((UncheckedList<object>)managedStack).RemoveRange(mStackBase, managedStack.Count - mStackBase);
- #endif
- valueTypePtr = frame.ValueTypeBasePointer;
- return ret;
- }
- public void RelocateValueTypeAndFreeAfterDst(StackObject* src, StackObject* dst)
- {
- var objRef2 = dst;
- dst = ILIntepreter.ResolveReference(dst);
- int start = int.MaxValue;
- int end = int.MaxValue;
- CountValueTypeManaged(objRef2, ref start, ref end, &objRef2);
- RelocateValueType(src, ref dst, ref start);
- ValueTypeStackPointer = dst;
- if (start <= end)
- RemoveManagedStackRange(start, end);
- }
- void RelocateValueType(StackObject* src, ref StackObject* dst, ref int mStackBase)
- {
- StackObject* descriptor = ILIntepreter.ResolveReference(src);
- if (descriptor > dst)
- throw new StackOverflowException();
- *dst = *descriptor;
- int cnt = descriptor->ValueLow;
- StackObject* endAddr = ILIntepreter.Minus(dst, cnt + 1);
- for(int i = 0; i < cnt; i++)
- {
- StackObject* addr = ILIntepreter.Minus(descriptor, i + 1);
- StackObject* tarVal = ILIntepreter.Minus(dst, i + 1);
- *tarVal = *addr;
- switch (addr->ObjectType)
- {
- case ObjectTypes.Object:
- case ObjectTypes.ArrayReference:
- case ObjectTypes.FieldReference:
- if (tarVal->Value >= mStackBase)
- {
- int oldIdx = addr->Value;
- tarVal->Value = mStackBase;
- managedStack[mStackBase] = managedStack[oldIdx];
- mStackBase++;
- }
- break;
- case ObjectTypes.ValueTypeObjectReference:
- var newAddr = endAddr;
- RelocateValueType(addr, ref endAddr, ref mStackBase);
- *(long*)&tarVal->Value = (long)newAddr;
- break;
- }
- }
- dst = endAddr;
- }
- public void AllocValueType(StackObject* ptr, IType type)
- {
- if (type.IsValueType)
- {
- int fieldCount = 0;
- if(type is ILType)
- {
- fieldCount = ((ILType)type).TotalFieldCount;
- }
- else
- {
- fieldCount = ((CLRType)type).TotalFieldCount;
- }
- ptr->ObjectType = ObjectTypes.ValueTypeObjectReference;
- var dst = valueTypePtr;
- *(long*)&ptr->Value = (long)dst;
- dst->ObjectType = ObjectTypes.ValueTypeDescriptor;
- dst->Value = type.GetHashCode();
- dst->ValueLow = fieldCount;
- valueTypePtr = ILIntepreter.Minus(valueTypePtr, fieldCount + 1);
- if (valueTypePtr <= StackBase)
- throw new StackOverflowException();
- InitializeValueTypeObject(type, dst);
- }
- else
- throw new ArgumentException(type.FullName + " is not a value type.", "type");
- }
- void InitializeValueTypeObject(IType type, StackObject* ptr)
- {
- if (type is ILType)
- {
- ILType t = (ILType)type;
- for (int i = 0; i < t.FieldTypes.Length; i++)
- {
- var ft = t.FieldTypes[i];
- StackObject* val = ILIntepreter.Minus(ptr, t.FieldStartIndex + i + 1);
- if (ft.IsPrimitive || ft.IsEnum)
- StackObject.Initialized(val, ft);
- else
- {
- if (ft.IsValueType)
- {
- if (ft is ILType || ((CLRType)ft).ValueTypeBinder != null)
- AllocValueType(val, ft);
- else
- {
- val->ObjectType = ObjectTypes.Object;
- val->Value = managedStack.Count;
- managedStack.Add(((CLRType)ft).CreateDefaultInstance());
- }
- }
- else
- {
- val->ObjectType = ObjectTypes.Object;
- val->Value = managedStack.Count;
- managedStack.Add(null);
- }
- }
- }
- if (type.BaseType != null && type.BaseType is ILType)
- InitializeValueTypeObject((ILType)type.BaseType, ptr);
- }
- else
- {
- CLRType t = (CLRType)type;
- var cnt = t.TotalFieldCount;
- for(int i = 0; i < cnt; i++)
- {
- var it = t.OrderedFieldTypes[i] as CLRType;
- StackObject* val = ILIntepreter.Minus(ptr, i + 1);
- if (it.IsPrimitive || it.IsEnum)
- StackObject.Initialized(val, it);
- else
- {
- if (it.IsValueType)
- {
- if (it.ValueTypeBinder != null)
- AllocValueType(val, it);
- else
- {
- val->ObjectType = ObjectTypes.Object;
- val->Value = managedStack.Count;
- managedStack.Add(it.CreateDefaultInstance());
- }
- }
- else
- {
- val->ObjectType = ObjectTypes.Object;
- val->Value = managedStack.Count;
- managedStack.Add(null);
- }
- }
- }
- }
- }
- public void ClearValueTypeObject(IType type, StackObject* ptr)
- {
- if (type is ILType)
- {
- ILType t = (ILType)type;
- for (int i = 0; i < t.FieldTypes.Length; i++)
- {
- var ft = t.FieldTypes[i];
- StackObject* val = ILIntepreter.Minus(ptr, t.FieldStartIndex + i + 1);
- if (ft.IsPrimitive || ft.IsEnum)
- StackObject.Initialized(val, ft);
- else
- {
- switch (val->ObjectType)
- {
- case ObjectTypes.ValueTypeObjectReference:
- ClearValueTypeObject(ft, ILIntepreter.ResolveReference(val));
- break;
- default:
- if (ft.IsValueType)
- {
- if(ft is ILType)
- {
- throw new NotImplementedException();
- }
- else
- {
- managedStack[val->Value] = ((CLRType)ft).CreateDefaultInstance();
- }
- }
- else
- managedStack[val->Value] = null;
- break;
- }
- }
- }
- if (type.BaseType != null && type.BaseType is ILType)
- ClearValueTypeObject((ILType)type.BaseType, ptr);
- }
- else
- {
- CLRType t = (CLRType)type;
- var cnt = t.TotalFieldCount;
- for (int i = 0; i < cnt; i++)
- {
- var vt = t.OrderedFieldTypes[i] as CLRType;
- StackObject* val = ILIntepreter.Minus(ptr, i + 1);
- if (vt.IsPrimitive)
- StackObject.Initialized(val, vt);
- else
- {
- switch (val->ObjectType)
- {
- case ObjectTypes.ValueTypeObjectReference:
- {
- var dst = ILIntepreter.ResolveReference(val);
- ClearValueTypeObject(vt, dst);
- }
- break;
- default:
- if (vt.IsValueType)
- {
- managedStack[val->Value] = vt.CreateDefaultInstance();
- }
- else
- managedStack[val->Value] = null;
- break;
- }
- }
- }
- }
- }
- void RemoveManagedStackRange(int start, int end)
- {
- if (start != int.MaxValue)
- {
- if (end == managedStack.Count - 1)
- {
- #if DEBUG && !DISABLE_ILRUNTIME_DEBUG
- ((List<object>)managedStack).RemoveRange(start, managedStack.Count - start);
- #else
- ((UncheckedList<object>)managedStack).RemoveRange(start, managedStack.Count - start);
- #endif
- }
- else
- throw new NotSupportedException();
- }
- }
- public void FreeValueTypeObject(StackObject* esp)
- {
- if (esp->ObjectType != ObjectTypes.ValueTypeObjectReference)
- return;
- int start = int.MaxValue;
- int end = int.MinValue;
- StackObject* endAddr;
- CountValueTypeManaged(esp, ref start, ref end, &endAddr);
- if (endAddr == valueTypePtr)
- valueTypePtr = ILIntepreter.ResolveReference(esp);
- else
- throw new NotSupportedException();
- RemoveManagedStackRange(start, end);
- }
- void CountValueTypeManaged(StackObject* esp, ref int start, ref int end, StackObject** endAddr)
- {
- StackObject* descriptor = ILIntepreter.ResolveReference(esp);
- int cnt = descriptor->ValueLow;
- *endAddr = ILIntepreter.Minus(descriptor, cnt + 1);
- for (int i = 0; i < cnt; i++)
- {
- StackObject* addr = ILIntepreter.Minus(descriptor, i + 1);
- switch (addr->ObjectType)
- {
- case ObjectTypes.Object:
- case ObjectTypes.ArrayReference:
- case ObjectTypes.FieldReference:
- {
- if (start == int.MaxValue)
- {
- start = addr->Value;
- end = start;
- }
- else if (addr->Value == end + 1)
- end++;
- else
- throw new NotSupportedException();
- }
- break;
- case ObjectTypes.ValueTypeObjectReference:
- CountValueTypeManaged(addr, ref start, ref end, endAddr);
- break;
- }
- }
- }
- public void Dispose()
- {
- if (nativePointer != IntPtr.Zero)
- {
- System.Runtime.InteropServices.Marshal.FreeHGlobal(nativePointer);
- nativePointer = IntPtr.Zero;
- }
- }
- StackObject* Add(StackObject* a, int b)
- {
- return (StackObject*)((long)a + sizeof(StackObject) * b);
- }
- }
- }
|