Files
CounterStrikeSharp/managed/CounterStrikeSharp.API/Core/ScriptContext.cs
2023-11-09 09:18:33 +10:00

492 lines
11 KiB
C#

/*
* Copyright (c) 2014 Bas Timmer/NTAuthority et al.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*
* This file has been modified from its original form for use in this program
* under GNU Lesser General Public License, version 2.
*/
using System;
using System.Collections.Concurrent;
using System.Runtime.InteropServices;
using System.Security;
using System.Text;
namespace CounterStrikeSharp.API.Core
{
public class NativeException : Exception
{
public NativeException(string message) : base(message)
{
}
}
[StructLayout(LayoutKind.Sequential)]
[Serializable]
public unsafe struct fxScriptContext
{
public int numArguments;
public int numResults;
public int hasError;
public ulong nativeIdentifier;
public fixed byte functionData[8 * 32];
public fixed byte result[8];
}
public class ScriptContext
{
[ThreadStatic]
private static ScriptContext _globalScriptContext;
public static ScriptContext GlobalScriptContext
{
get
{
if (_globalScriptContext == null) _globalScriptContext = new ScriptContext();
return _globalScriptContext;
}
}
public unsafe ScriptContext()
{
//Console.WriteLine("Global context address: " + (IntPtr)m_extContext);
}
public unsafe ScriptContext(fxScriptContext* context)
{
m_extContext = *context;
//Console.WriteLine("Global context address: " + (IntPtr)m_extContext);
}
private readonly ConcurrentQueue<Action> ms_finalizers = new ConcurrentQueue<Action>();
private readonly object ms_lock = new object();
internal object Lock => ms_lock;
internal fxScriptContext m_extContext = new fxScriptContext();
[SecuritySafeCritical]
public void Reset()
{
InternalReset();
}
[SecurityCritical]
private void InternalReset()
{
m_extContext.numArguments = 0;
m_extContext.numResults = 0;
m_extContext.hasError = 0;
//CleanUp();
}
[SecuritySafeCritical]
public void Invoke()
{
InvokeNativeInternal();
GlobalCleanUp();
}
[SecurityCritical]
private void InvokeNativeInternal()
{
unsafe
{
fixed (fxScriptContext* cxt = &m_extContext)
{
Helpers.InvokeNative(new IntPtr(cxt));
}
}
}
public unsafe byte[] GetBytes()
{
fixed (fxScriptContext* context = &m_extContext)
{
byte[] arr = new byte[8 * 32];
Marshal.Copy((IntPtr)context->functionData, arr, 0, 8 * 32);
return arr;
}
}
public unsafe IntPtr GetContextUnderlyingAddress()
{
fixed (fxScriptContext* context = &m_extContext)
{
return (IntPtr)context;
}
}
[SecuritySafeCritical]
public void Push(object arg)
{
PushInternal(arg);
}
[SecuritySafeCritical]
public unsafe void SetResult(object arg, fxScriptContext* cxt)
{
SetResultInternal(cxt, arg);
}
[SecurityCritical]
private unsafe void PushInternal(object arg)
{
fixed (fxScriptContext* context = &m_extContext)
{
Push(context, arg);
}
}
[SecurityCritical]
public unsafe void SetIdentifier(ulong arg)
{
fixed (fxScriptContext* context = &m_extContext)
{
context->nativeIdentifier = arg;
}
}
public unsafe void CheckErrors()
{
fixed (fxScriptContext* context = &m_extContext)
{
if (Convert.ToBoolean(context->hasError))
{
string error = GetResult<string>();
Reset();
throw new NativeException(error);
}
}
}
[SecurityCritical]
internal unsafe void Push(fxScriptContext* context, object arg)
{
if (arg == null)
{
arg = 0;
}
if (arg.GetType().IsEnum)
{
arg = Convert.ChangeType(arg, arg.GetType().GetEnumUnderlyingType());
}
if (arg is string)
{
var str = (string)Convert.ChangeType(arg, typeof(string));
PushString(context, str);
return;
}
else if (arg is InputArgument ia)
{
Push(context, ia.Value);
return;
}
if (Marshal.SizeOf(arg.GetType()) <= 8)
{
PushUnsafe(context, arg);
}
context->numArguments++;
}
[SecurityCritical]
internal unsafe void SetResultInternal(fxScriptContext* context, object arg)
{
if (arg == null)
{
arg = 0;
}
if (arg.GetType().IsEnum)
{
arg = Convert.ChangeType(arg, arg.GetType().GetEnumUnderlyingType());
}
if (arg is string)
{
var str = (string)Convert.ChangeType(arg, typeof(string));
SetResultString(context, str);
return;
}
else if (arg is InputArgument ia)
{
SetResultInternal(context, ia.Value);
return;
}
if (Marshal.SizeOf(arg.GetType()) <= 8)
{
SetResultUnsafe(context, arg);
}
}
[SecurityCritical]
internal unsafe void PushUnsafe(fxScriptContext* cxt, object arg)
{
*(long*)(&cxt->functionData[8 * cxt->numArguments]) = 0;
Marshal.StructureToPtr(arg, new IntPtr(cxt->functionData + (8 * cxt->numArguments)), true);
}
[SecurityCritical]
internal unsafe void SetResultUnsafe(fxScriptContext* cxt, object arg)
{
*(long*)(&cxt->result[0]) = 0;
Marshal.StructureToPtr(arg, new IntPtr(cxt->result), true);
}
[SecurityCritical]
internal unsafe void PushString(string str)
{
fixed (fxScriptContext* cxt = &m_extContext)
{
PushString(cxt, str);
}
}
[SecurityCritical]
internal unsafe void PushString(fxScriptContext* cxt, string str)
{
var ptr = IntPtr.Zero;
if (str != null)
{
var b = Encoding.UTF8.GetBytes(str);
ptr = Marshal.AllocHGlobal(b.Length + 1);
Marshal.Copy(b, 0, ptr, b.Length);
Marshal.WriteByte(ptr, b.Length, 0);
ms_finalizers.Enqueue(() => Free(ptr));
}
unsafe
{
*(IntPtr*)(&cxt->functionData[8 * cxt->numArguments]) = ptr;
}
cxt->numArguments++;
}
[SecurityCritical]
internal unsafe void SetResultString(fxScriptContext* cxt, string str)
{
var ptr = IntPtr.Zero;
if (str != null)
{
var b = Encoding.UTF8.GetBytes(str);
ptr = Marshal.AllocHGlobal(b.Length + 1);
Marshal.Copy(b, 0, ptr, b.Length);
Marshal.WriteByte(ptr, b.Length, 0);
ms_finalizers.Enqueue(() => Free(ptr));
}
unsafe
{
*(IntPtr*)(&cxt->result[8]) = ptr;
}
}
[SecuritySafeCritical]
private void Free(IntPtr ptr)
{
Marshal.FreeHGlobal(ptr);
}
[SecuritySafeCritical]
public T GetArgument<T>(int index)
{
return (T)GetArgument(typeof(T), index);
}
[SecuritySafeCritical]
public object GetArgument(Type type, int index)
{
return GetArgumentHelper(type, index);
}
[SecurityCritical]
internal unsafe object GetArgument(fxScriptContext* cxt, Type type, int index)
{
return GetArgumentHelper(cxt, type, index);
}
[SecurityCritical]
private unsafe object GetArgumentHelper(Type type, int index)
{
fixed (fxScriptContext* cxt = &m_extContext)
{
return GetArgumentHelper(cxt, type, index);
}
}
[SecurityCritical]
private unsafe object GetArgumentHelper(fxScriptContext* context, Type type, int index)
{
return GetResult(type, &context->functionData[index * 8]);
}
[SecuritySafeCritical]
public T GetResult<T>()
{
return (T)GetResult(typeof(T));
}
[SecuritySafeCritical]
public object GetResult(Type type)
{
return GetResultHelper(type);
}
[SecurityCritical]
internal unsafe object GetResult(fxScriptContext* cxt, Type type)
{
return GetResultHelper(cxt, type);
}
[SecurityCritical]
private unsafe object GetResultHelper(Type type)
{
fixed (fxScriptContext* cxt = &m_extContext)
{
return GetResultHelper(cxt, type);
}
}
[SecurityCritical]
private unsafe object GetResultHelper(fxScriptContext* context, Type type)
{
return GetResult(type, &context->result[0]);
}
[SecurityCritical]
internal unsafe object GetResult(Type type, byte* ptr)
{
if (type == typeof(string))
{
var nativeUtf8 = *(IntPtr*)&ptr[0];
if (nativeUtf8 == IntPtr.Zero)
{
return null;
}
var len = 0;
while (Marshal.ReadByte(nativeUtf8, len) != 0)
{
++len;
}
var buffer = new byte[len];
Marshal.Copy(nativeUtf8, buffer, 0, buffer.Length);
return Encoding.UTF8.GetString(buffer);
}
if (typeof(NativeObject).IsAssignableFrom(type))
{
var pointer = (IntPtr)GetResult(typeof(IntPtr), ptr);
return Activator.CreateInstance(type, pointer);
}
if (type == typeof(object))
{
// var dataPtr = *(IntPtr*)&ptr[0];
// var dataLength = *(long*)&ptr[8];
//
// byte[] data = new byte[dataLength];
// Marshal.Copy(dataPtr, data, 0, (int)dataLength);
return null;
//return MsgPackDeserializer.Deserialize(data);
}
if (type.IsEnum)
{
return Enum.ToObject(type, (int)GetResult(typeof(int), ptr));
}
if (Marshal.SizeOf(type) <= 8)
{
return GetResultInternal(type, ptr);
}
return null;
}
[SecurityCritical]
private unsafe object GetResultInternal(Type type, byte* ptr)
{
var obj = Marshal.PtrToStructure(new IntPtr(ptr), type);
return obj;
}
[SecurityCritical]
internal unsafe string ErrorHandler(byte* error)
{
if (error != null)
{
var errorStart = error;
int length = 0;
for (var p = errorStart; *p != 0; p++)
{
length++;
}
return Encoding.UTF8.GetString(errorStart, length);
}
return "Native invocation failed.";
}
internal void GlobalCleanUp()
{
lock (ms_lock)
{
while (ms_finalizers.TryDequeue(out var cb))
{
cb();
}
}
}
public override string ToString()
{
return $"ScriptContext{{numArgs={m_extContext.numArguments}}}";
}
}
}