#include "il2cpp-config.h" #include "vm/Assembly.h" #include "vm/AssemblyName.h" #include "vm/MetadataCache.h" #include "vm/Runtime.h" #include "vm-utils/VmStringUtils.h" #include "il2cpp-tabledefs.h" #include "il2cpp-class-internals.h" #include "Baselib.h" #include "Cpp/ReentrantLock.h" #include "os/Atomic.h" #include #include namespace il2cpp { namespace vm { static baselib::ReentrantLock s_assemblyLock; // copy on write static int32_t s_assemblyVersion = 0; static AssemblyVector s_Assemblies; static int32_t s_snapshotAssemblyVersion = 0; static AssemblyVector* s_snapshotAssemblies = &s_Assemblies; static void CopyValidAssemblies(AssemblyVector& dst, const AssemblyVector& src) { for (AssemblyVector::const_iterator assIt = src.begin(); assIt != src.end(); ++assIt) { const Il2CppAssembly* ass = *assIt; if (ass->token) { dst.push_back(ass); } } } AssemblyVector* Assembly::GetAllAssemblies() { os::FastAutoLock lock(&s_assemblyLock); if (s_assemblyVersion != s_snapshotAssemblyVersion) { s_snapshotAssemblies = new AssemblyVector(); CopyValidAssemblies(*s_snapshotAssemblies, s_Assemblies); s_snapshotAssemblyVersion = s_assemblyVersion; } return s_snapshotAssemblies; } void Assembly::GetAllAssemblies(AssemblyVector& assemblies) { os::FastAutoLock lock(&s_assemblyLock); if (s_assemblyVersion != s_snapshotAssemblyVersion) { CopyValidAssemblies(assemblies, s_Assemblies); } else { assemblies = *s_snapshotAssemblies; } } const Il2CppAssembly* Assembly::GetLoadedAssembly(const char* name) { os::FastAutoLock lock(&s_assemblyLock); AssemblyVector& assemblies = s_Assemblies; for (AssemblyVector::const_reverse_iterator assembly = assemblies.rbegin(); assembly != assemblies.rend(); ++assembly) { if (strcmp((*assembly)->aname.name, name) == 0) return *assembly; } return NULL; } Il2CppImage* Assembly::GetImage(const Il2CppAssembly* assembly) { return assembly->image; } void Assembly::GetReferencedAssemblies(const Il2CppAssembly* assembly, AssemblyNameVector* target) { for (int32_t sourceIndex = 0; sourceIndex < assembly->referencedAssemblyCount; sourceIndex++) { const Il2CppAssembly* refAssembly = MetadataCache::GetReferencedAssembly(assembly, sourceIndex); target->push_back(&refAssembly->aname); } } static bool ends_with(const char *str, const char *suffix) { if (!str || !suffix) return false; const size_t lenstr = strlen(str); const size_t lensuffix = strlen(suffix); if (lensuffix > lenstr) return false; return strncmp(str + lenstr - lensuffix, suffix, lensuffix) == 0; } const Il2CppAssembly* Assembly::Load(const char* name) { const Il2CppAssembly* loadedAssembly = MetadataCache::GetAssemblyByName(name); if (loadedAssembly) { return loadedAssembly; } if (!ends_with(name, ".dll") && !ends_with(name, ".exe")) { const size_t len = strlen(name); char *tmp = new char[len + 5]; memset(tmp, 0, len + 5); memcpy(tmp, name, len); memcpy(tmp + len, ".dll", 4); loadedAssembly = MetadataCache::GetAssemblyByName(tmp); if (!loadedAssembly) { memcpy(tmp + len, ".exe", 4); loadedAssembly = MetadataCache::GetAssemblyByName(tmp); } delete[] tmp; return loadedAssembly; } else { return nullptr; } } void Assembly::Register(const Il2CppAssembly* assembly) { os::FastAutoLock lock(&s_assemblyLock); s_Assemblies.push_back(assembly); ++s_assemblyVersion; } void Assembly::InvalidateAssemblyList() { os::FastAutoLock lock(&s_assemblyLock); ++s_assemblyVersion; } void Assembly::ClearAllAssemblies() { os::FastAutoLock lock(&s_assemblyLock); s_Assemblies.clear(); delete s_snapshotAssemblies; s_snapshotAssemblies = &s_Assemblies; s_assemblyVersion = s_snapshotAssemblyVersion = 0; } void Assembly::Initialize() { } } /* namespace vm */ } /* namespace il2cpp */