1.前言
存活标记,意即当进行垃圾回收的时候被存活对象需要进行标记。固定对象即在GC堆不能被移动的对象,它也是通过标记来标注它是否是固定对象。本篇来看下它们是如何标记的。
2.参考
固定对象可以参考:.Net8罕见的技术:固定对象的操作
垃圾回收GCinfo参考:GC垃圾回收的GCInfo编码过程
3.概述
一.设置存活对象:
void SetMarked()
{
_ASSERTE(RawGetMethodTable());
RawSetMethodTable((MethodTable *) (((size_t) RawGetMethodTable()) | GC_MARKED));
}
很好理解,把MethodTable最后一位设置为1,即此对象存活。
二.获取对象是否存活
BOOL IsMarked() const
{
return !!(((size_t)RawGetMethodTable()) & GC_MARKED);
}
也很简单,看MethodTable最后一位是否为1
三.设置对象为固定对象
void SetPinned()
{
assert (!(gc_heap::settings.concurrent));
GetHeader()->SetGCBit();
}
看到GetHeader()是获取对象头,SetGCBit里面是跟对象头或(|)0x20000000.
也即对象头的高三位设置为1.
四.判断对象是否固定对象
BOOL IsPinned() const
{
return !!((((CObjectHeader*)this)->GetHeader()->GetBits()) & BIT_SBLK_GC_RESERVE);
}
判断对象头的高三位是否为1就行了
五.清除标记
void ClearMarked()
{
#ifdef DOUBLY_LINKED_FL
RawSetMethodTable ((MethodTable *)(((size_t) RawGetMethodTable()) & (~GC_MARKED)));
#else
RawSetMethodTable (GetMethodTable());
#endif //DOUBLY_LINKED_FL
}
把MethodTable最后一位设置为0,即可。