DeF*_*zer 5 c# reflection pointers
如何在 C# 中找到指向托管类的原始指针,并且希望它是内存中的原始大小?显然,这是 CLR 不允许的——更准确地说,是严格禁止的,因为出于稳定性和安全的原因,永远不应该使用托管类的非托管表示形式——所以我正在寻找一个黑客。我不是在寻找序列化 - 我实际上需要一个托管类的转储,因为它在原始内存中表示。
更准确地说,我正在getObjectPtr以下示例中寻找类似函数的东西:
IntPtr getObjectPtr(Object managedClass) {...}
void main() {
var test=new TestClass();
IntPtr* ptr_to_test=getObjectPtr(test);
Console.WriteLine(ptr_to_test.ToString());
}
Run Code Online (Sandbox Code Playgroud)
提前致谢!
编辑: 我终于自己找到了一个解决方案,当我回来将其作为答案发布时,对如此快速发布的答案数量感到非常惊讶......谢谢大家!这非常快,而且完全出乎意料。
最接近我的解决方案是@thehennyy 的解决方案,但我没有发布它,因为@Chino 提出了更好的解决方案(对不起,我一开始误认为它是错误的,我只是忘记再次取消引用指针)。它不需要不安全的代码,并且更能容忍 GC:
class Program
{
// Here is the function in case anyone needs it.
// Note, though, it does not preserve the handle while you work with
// pointer, so it is less reliable than the code in Main():
static IntPtr getPointerToObject(Object unmanagedObject)
{
GCHandle gcHandle = GCHandle.Alloc(unmanagedObject, GCHandleType.WeakTrackResurrection);
IntPtr thePointer = Marshal.ReadIntPtr(GCHandle.ToIntPtr(gcHandle));
gcHandle.Free();
return thePointer;
}
class TestClass
{
uint a = 0xDEADBEEF;
}
static void Main(string[] args)
{
byte[] cls = new byte[16];
var test = new TestClass();
GCHandle gcHandle = GCHandle.Alloc(test, GCHandleType.WeakTrackResurrection);
IntPtr thePointer = Marshal.ReadIntPtr(GCHandle.ToIntPtr(gcHandle));
Marshal.Copy(thePointer, cls, 0, 16); //Dump first 16 bytes...
Console.WriteLine(BitConverter.ToString(BitConverter.GetBytes(thePointer.ToInt32())));
Console.WriteLine(BitConverter.ToString(cls));
Console.ReadLine();
gcHandle.Free();
}
}
/* Example output (yours should be different):
40-23-CA-02
4C-38-04-01-EF-BE-AD-DE-00-00-00-80-B4-21-50-73
That field's value is "EF-BE-AD-DE", 0xDEADBEEF as it is stored in memory. Yay, we found it!
*/
Run Code Online (Sandbox Code Playgroud)
然而,现在我有点不知所措。根据这篇文章,类中的前2个地址应该是指向SyncBlock和RTTI结构的指针,因此第一个字段的地址必须偏移2个字[32位系统8个字节,64位系统16个字节]从一开始就。我的是 64 位的;但是,正如您在输出中看到的,很明显第一个字段相对于对象地址的原始偏移量只有 4 个字节,这没有任何意义。
嘿,这就是你想要的吗?:
GCHandle gcHandle = GCHandle.Alloc(yourObject,GCHandleType.WeakTrackResurrection);
IntPtr thePointer = GCHandle.ToIntPtr(gcHandle);
Run Code Online (Sandbox Code Playgroud)
您可以编写一个泄漏对象地址的小型 IL 函数。
var o = new object();
var d = new DynamicMethod("GetPtr", typeof(IntPtr), new Type[] {typeof(object)}, Assembly.GetExecutingAssembly().ManifestModule);
var il = d.GetILGenerator();
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Ret);
var address = (IntPtr)d.Invoke(null, new object[] {o});
Console.WriteLine(address);
Run Code Online (Sandbox Code Playgroud)
来源是:IllidanS4 / SharpUtils / UnsafeTools.cs
| 归档时间: |
|
| 查看次数: |
2539 次 |
| 最近记录: |