#include "il2cpp-config.h"
|
|
#if IL2CPP_GC_BOEHM
|
|
#include <stdint.h>
|
#include "gc_wrapper.h"
|
#include "GarbageCollector.h"
|
#include "WriteBarrierValidation.h"
|
#include "vm/Profiler.h"
|
#include "utils/Il2CppHashMap.h"
|
#include "utils/HashUtils.h"
|
|
static bool s_GCInitialized = false;
|
|
#if IL2CPP_ENABLE_DEFERRED_GC
|
static bool s_PendingGC = false;
|
#endif
|
|
#if IL2CPP_ENABLE_PROFILER
|
using il2cpp::vm::Profiler;
|
static void on_gc_event(GC_EventType eventType);
|
static void on_heap_resize(GC_word newSize);
|
#endif
|
|
#if !IL2CPP_TINY_WITHOUT_DEBUGGER
|
static GC_push_other_roots_proc default_push_other_roots;
|
typedef Il2CppHashMap<char*, char*, il2cpp::utils::PassThroughHash<char*> > RootMap;
|
static RootMap s_Roots;
|
|
typedef Il2CppHashMap<void*, il2cpp::gc::GarbageCollector::GetDynamicRootDataProc, il2cpp::utils::PassThroughHash<void*> > DynamicRootMap;
|
static DynamicRootMap s_DynamicRoots;
|
|
static void push_other_roots(void);
|
#endif // !IL2CPP_TINY_WITHOUT_DEBUGGER
|
|
void
|
il2cpp::gc::GarbageCollector::Initialize()
|
{
|
if (s_GCInitialized)
|
return;
|
|
#if IL2CPP_ENABLE_WRITE_BARRIER_VALIDATION
|
il2cpp::gc::WriteBarrierValidation::Setup();
|
#endif
|
// This tells the GC that we are not scanning dynamic library data segments and that
|
// the GC tracked data structures need ot be manually pushed and marked.
|
// Call this before GC_INIT since the initialization logic uses this value.
|
GC_set_no_dls(1);
|
|
#if !IL2CPP_DEVELOPMENT
|
// Turn off GC logging and warnings for non-development builds
|
GC_set_warn_proc(GC_ignore_warn_proc);
|
#endif
|
|
#if IL2CPP_ENABLE_WRITE_BARRIERS
|
GC_enable_incremental();
|
#if IL2CPP_INCREMENTAL_TIME_SLICE
|
GC_set_time_limit(IL2CPP_INCREMENTAL_TIME_SLICE);
|
#endif
|
#endif
|
|
#if !IL2CPP_TINY_WITHOUT_DEBUGGER
|
default_push_other_roots = GC_get_push_other_roots();
|
GC_set_push_other_roots(push_other_roots);
|
#endif // !IL2CPP_TINY_WITHOUT_DEBUGGER
|
|
#if IL2CPP_ENABLE_PROFILER
|
GC_set_on_collection_event(&on_gc_event);
|
GC_set_on_heap_resize(&on_heap_resize);
|
#endif
|
|
GC_INIT();
|
#if defined(GC_THREADS)
|
GC_set_finalize_on_demand(1);
|
#if !IL2CPP_TINY_WITHOUT_DEBUGGER
|
GC_set_finalizer_notifier(&il2cpp::gc::GarbageCollector::NotifyFinalizers);
|
#endif
|
// We need to call this if we want to manually register threads, i.e. GC_register_my_thread
|
#if !IL2CPP_TARGET_JAVASCRIPT
|
GC_allow_register_threads();
|
#endif
|
#endif
|
#ifdef GC_GCJ_SUPPORT
|
GC_init_gcj_malloc(0, NULL);
|
#endif
|
s_GCInitialized = true;
|
}
|
|
void il2cpp::gc::GarbageCollector::UninitializeGC()
|
{
|
#if IL2CPP_ENABLE_WRITE_BARRIER_VALIDATION
|
il2cpp::gc::WriteBarrierValidation::Run();
|
#endif
|
GC_deinit();
|
}
|
|
int32_t
|
il2cpp::gc::GarbageCollector::GetCollectionCount(int32_t generation)
|
{
|
return (int32_t)GC_get_gc_no();
|
}
|
|
int32_t
|
il2cpp::gc::GarbageCollector::GetMaxGeneration()
|
{
|
return 0;
|
}
|
|
void
|
il2cpp::gc::GarbageCollector::Collect(int maxGeneration)
|
{
|
#if IL2CPP_ENABLE_DEFERRED_GC
|
if (GC_is_disabled())
|
s_PendingGC = true;
|
#endif
|
GC_gcollect();
|
}
|
|
int32_t
|
il2cpp::gc::GarbageCollector::CollectALittle()
|
{
|
#if IL2CPP_ENABLE_DEFERRED_GC
|
if (s_PendingGC)
|
{
|
s_PendingGC = false;
|
GC_gcollect();
|
return 0; // no more work to do
|
}
|
else
|
{
|
return GC_collect_a_little();
|
}
|
#else
|
return GC_collect_a_little();
|
#endif
|
}
|
|
#if IL2CPP_ENABLE_WRITE_BARRIERS
|
void
|
il2cpp::gc::GarbageCollector::SetWriteBarrier(void **ptr)
|
{
|
GC_END_STUBBORN_CHANGE(ptr);
|
}
|
|
#endif
|
|
int64_t
|
il2cpp::gc::GarbageCollector::GetUsedHeapSize(void)
|
{
|
return GC_get_heap_size() - GC_get_free_bytes();
|
}
|
|
int64_t
|
il2cpp::gc::GarbageCollector::GetAllocatedHeapSize(void)
|
{
|
return GC_get_heap_size();
|
}
|
|
void
|
il2cpp::gc::GarbageCollector::Disable()
|
{
|
GC_disable();
|
}
|
|
void
|
il2cpp::gc::GarbageCollector::Enable()
|
{
|
GC_enable();
|
}
|
|
bool
|
il2cpp::gc::GarbageCollector::IsDisabled()
|
{
|
return GC_is_disabled();
|
}
|
|
bool
|
il2cpp::gc::GarbageCollector::RegisterThread(void *baseptr)
|
{
|
#if defined(GC_THREADS) && !IL2CPP_TARGET_JAVASCRIPT
|
struct GC_stack_base sb;
|
int res;
|
|
res = GC_get_stack_base(&sb);
|
if (res != GC_SUCCESS)
|
{
|
sb.mem_base = baseptr;
|
#ifdef __ia64__
|
/* Can't determine the register stack bounds */
|
IL2CPP_ASSERT(false && "mono_gc_register_thread failed ().");
|
#endif
|
}
|
res = GC_register_my_thread(&sb);
|
if ((res != GC_SUCCESS) && (res != GC_DUPLICATE))
|
{
|
IL2CPP_ASSERT(false && "GC_register_my_thread () failed.");
|
return false;
|
}
|
#endif
|
return true;
|
}
|
|
bool
|
il2cpp::gc::GarbageCollector::UnregisterThread()
|
{
|
#if defined(GC_THREADS) && !IL2CPP_TARGET_JAVASCRIPT
|
int res;
|
|
res = GC_unregister_my_thread();
|
if (res != GC_SUCCESS)
|
IL2CPP_ASSERT(false && "GC_unregister_my_thread () failed.");
|
|
return res == GC_SUCCESS;
|
#else
|
return true;
|
#endif
|
}
|
|
il2cpp::gc::GarbageCollector::FinalizerCallback il2cpp::gc::GarbageCollector::RegisterFinalizerWithCallback(Il2CppObject* obj, FinalizerCallback callback)
|
{
|
FinalizerCallback oldCallback;
|
void* oldData;
|
GC_REGISTER_FINALIZER_NO_ORDER((char*)obj, callback, NULL, &oldCallback, &oldData);
|
IL2CPP_ASSERT(oldData == NULL);
|
return oldCallback;
|
}
|
|
void
|
il2cpp::gc::GarbageCollector::AddWeakLink(void **link_addr, Il2CppObject *obj, bool track)
|
{
|
/* libgc requires that we use HIDE_POINTER... */
|
*link_addr = (void*)GC_HIDE_POINTER(obj);
|
// need this since our strings are not real objects
|
if (GC_is_heap_ptr(obj))
|
GC_GENERAL_REGISTER_DISAPPEARING_LINK(link_addr, obj);
|
}
|
|
void
|
il2cpp::gc::GarbageCollector::RemoveWeakLink(void **link_addr)
|
{
|
Il2CppObject* obj = GarbageCollector::GetWeakLink(link_addr);
|
if (GC_is_heap_ptr(obj))
|
GC_unregister_disappearing_link(link_addr);
|
*link_addr = NULL;
|
}
|
|
static void*
|
RevealLink(void* link_addr)
|
{
|
void **link_a = (void**)link_addr;
|
return GC_REVEAL_POINTER(*link_a);
|
}
|
|
Il2CppObject*
|
il2cpp::gc::GarbageCollector::GetWeakLink(void **link_addr)
|
{
|
Il2CppObject *obj = (Il2CppObject*)GC_call_with_alloc_lock(RevealLink, link_addr);
|
if (obj == (Il2CppObject*)-1)
|
return NULL;
|
return obj;
|
}
|
|
void*
|
il2cpp::gc::GarbageCollector::MakeDescriptorForObject(size_t *bitmap, int numbits)
|
{
|
#ifdef GC_GCJ_SUPPORT
|
/* It seems there are issues when the bitmap doesn't fit: play it safe */
|
if (numbits >= 30)
|
return GC_NO_DESCRIPTOR;
|
else
|
return (void*)GC_make_descriptor((GC_bitmap)bitmap, numbits);
|
#else
|
return 0;
|
#endif
|
}
|
|
void* il2cpp::gc::GarbageCollector::MakeDescriptorForString()
|
{
|
return GC_NO_DESCRIPTOR;
|
}
|
|
void* il2cpp::gc::GarbageCollector::MakeDescriptorForArray()
|
{
|
return GC_NO_DESCRIPTOR;
|
}
|
|
void il2cpp::gc::GarbageCollector::StopWorld()
|
{
|
GC_stop_world_external();
|
}
|
|
void il2cpp::gc::GarbageCollector::StartWorld()
|
{
|
GC_start_world_external();
|
}
|
|
#if IL2CPP_TINY_WITHOUT_DEBUGGER
|
void*
|
il2cpp::gc::GarbageCollector::Allocate(size_t size)
|
{
|
return GC_MALLOC(size);
|
}
|
|
#endif
|
|
void*
|
il2cpp::gc::GarbageCollector::AllocateFixed(size_t size, void *descr)
|
{
|
// Note that we changed the implementation from mono.
|
// In our case, we expect that
|
// a) This memory will never be moved
|
// b) This memory will be scanned for references
|
// c) This memory will remain 'alive' until explicitly freed
|
// GC_MALLOC_UNCOLLECTABLE fulfills all these requirements
|
// It does not accept a descriptor, but there was only one
|
// or two places in mono that pass a descriptor to this routine
|
// and we can or will support those use cases in a different manner.
|
IL2CPP_ASSERT(!descr);
|
|
return GC_MALLOC_UNCOLLECTABLE(size);
|
}
|
|
void
|
il2cpp::gc::GarbageCollector::FreeFixed(void* addr)
|
{
|
GC_FREE(addr);
|
}
|
|
#if !IL2CPP_TINY_WITHOUT_DEBUGGER
|
int32_t
|
il2cpp::gc::GarbageCollector::InvokeFinalizers()
|
{
|
#if IL2CPP_TINY
|
return 0; // The Tiny profile does not have finalizers
|
#else
|
return (int32_t)GC_invoke_finalizers();
|
#endif
|
}
|
|
bool
|
il2cpp::gc::GarbageCollector::HasPendingFinalizers()
|
{
|
return GC_should_invoke_finalizers() != 0;
|
}
|
|
#endif
|
|
int64_t
|
il2cpp::gc::GarbageCollector::GetMaxTimeSliceNs()
|
{
|
return GC_get_time_limit_ns();
|
}
|
|
void
|
il2cpp::gc::GarbageCollector::SetMaxTimeSliceNs(int64_t maxTimeSlice)
|
{
|
GC_set_time_limit_ns(maxTimeSlice);
|
}
|
|
bool
|
il2cpp::gc::GarbageCollector::IsIncremental()
|
{
|
return GC_is_incremental_mode();
|
}
|
|
#if IL2CPP_ENABLE_PROFILER
|
|
void on_gc_event(GC_EventType eventType)
|
{
|
Profiler::GCEvent((Il2CppGCEvent)eventType);
|
}
|
|
void on_heap_resize(GC_word newSize)
|
{
|
Profiler::GCHeapResize((int64_t)newSize);
|
}
|
|
#endif // IL2CPP_ENABLE_PROFILER
|
|
void il2cpp::gc::GarbageCollector::ForEachHeapSection(void* user_data, HeapSectionCallback callback)
|
{
|
GC_foreach_heap_section(user_data, callback);
|
}
|
|
size_t il2cpp::gc::GarbageCollector::GetSectionCount()
|
{
|
return GC_get_heap_section_count();
|
}
|
|
void* il2cpp::gc::GarbageCollector::CallWithAllocLockHeld(GCCallWithAllocLockCallback callback, void* user_data)
|
{
|
return GC_call_with_alloc_lock(callback, user_data);
|
}
|
|
typedef struct
|
{
|
char *start;
|
char *end;
|
} RootData;
|
|
#if !IL2CPP_TINY_WITHOUT_DEBUGGER
|
|
static void*
|
register_root(void* arg)
|
{
|
RootData* root_data = (RootData*)arg;
|
s_Roots.insert(std::make_pair(root_data->start, root_data->end));
|
return NULL;
|
}
|
|
void il2cpp::gc::GarbageCollector::RegisterRoot(char *start, size_t size)
|
{
|
RootData root_data;
|
root_data.start = start;
|
/* Boehm root processing requires one byte past end of region to be scanned */
|
root_data.end = start + size + 1;
|
CallWithAllocLockHeld(register_root, &root_data);
|
}
|
|
static void*
|
deregister_root(void* arg)
|
{
|
s_Roots.erase((char*)arg);
|
return NULL;
|
}
|
|
void il2cpp::gc::GarbageCollector::UnregisterRoot(char* start)
|
{
|
GC_call_with_alloc_lock(deregister_root, start);
|
}
|
|
struct DynamicRootData
|
{
|
void* root;
|
il2cpp::gc::GarbageCollector::GetDynamicRootDataProc getRootDataFunc;
|
};
|
|
static void* register_dynamic_root(void* arg)
|
{
|
DynamicRootData* rootData = (DynamicRootData*)arg;
|
IL2CPP_ASSERT(s_DynamicRoots.find(rootData->root) == s_DynamicRoots.end());
|
s_DynamicRoots.add(rootData->root, rootData->getRootDataFunc);
|
|
return NULL;
|
}
|
|
static void* deregister_dynamic_root(void* arg)
|
{
|
IL2CPP_ASSERT(s_DynamicRoots.find(arg) != s_DynamicRoots.end());
|
s_DynamicRoots.erase(arg);
|
return NULL;
|
}
|
|
void il2cpp::gc::GarbageCollector::RegisterDynamicRoot(void* root, GetDynamicRootDataProc getRootDataFunc)
|
{
|
DynamicRootData rootData = {root, getRootDataFunc};
|
GC_call_with_alloc_lock(register_dynamic_root, &rootData);
|
}
|
|
void il2cpp::gc::GarbageCollector::UnregisterDynamicRoot(void* root)
|
{
|
GC_call_with_alloc_lock(deregister_dynamic_root, root);
|
}
|
|
static void
|
push_other_roots(void)
|
{
|
for (RootMap::iterator iter = s_Roots.begin(); iter != s_Roots.end(); ++iter)
|
GC_push_all(iter->first, iter->second);
|
for (auto dynamicRootEntry : s_DynamicRoots)
|
{
|
std::pair<char*, size_t> dynamicRootData = dynamicRootEntry.second(dynamicRootEntry.first);
|
if (dynamicRootData.first)
|
{
|
GC_push_all(dynamicRootData.first, dynamicRootData.first + dynamicRootData.second);
|
}
|
}
|
if (default_push_other_roots)
|
default_push_other_roots();
|
}
|
|
#endif // !IL2CPP_TINY_WITHOUT_DEBUGGER
|
|
#endif
|