Image.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. #include "il2cpp-config.h"
  2. #include <vector>
  3. #include <map>
  4. #include <limits>
  5. #include "os/Mutex.h"
  6. #include "utils/StringUtils.h"
  7. #include "vm/Array.h"
  8. #include "vm/Class.h"
  9. #include "vm/Image.h"
  10. #include "vm/MetadataCache.h"
  11. #include "vm/Reflection.h"
  12. #include "vm/StackTrace.h"
  13. #include "vm/Type.h"
  14. #include "utils/HashUtils.h"
  15. #include "utils/Il2CppHashMap.h"
  16. #include "utils/MemoryMappedFile.h"
  17. #include "utils/StringUtils.h"
  18. #include "vm-utils/VmStringUtils.h"
  19. #include "Baselib.h"
  20. #include "Cpp/ReentrantLock.h"
  21. #include "os/Atomic.h"
  22. #include "hybridclr/metadata/Image.h"
  23. struct NamespaceAndNamePairHash
  24. {
  25. size_t operator()(const std::pair<const char*, const char*>& pair) const
  26. {
  27. return il2cpp::utils::HashUtils::Combine(il2cpp::utils::StringUtils::Hash(pair.first), il2cpp::utils::StringUtils::Hash(pair.second));
  28. }
  29. };
  30. struct NamespaceAndNamePairEquals
  31. {
  32. bool operator()(const std::pair<const char*, const char*>& p1, const std::pair<const char*, const char*>& p2) const
  33. {
  34. return !strcmp(p1.first, p2.first) && !strcmp(p1.second, p2.second);
  35. }
  36. };
  37. struct NamespaceAndNamePairLess
  38. {
  39. bool operator()(const std::pair<const char*, const char*>& p1, const std::pair<const char*, const char*>& p2) const
  40. {
  41. int namespaceCompare = strcmp(p1.first, p2.first);
  42. if (namespaceCompare < 0)
  43. return true;
  44. if (namespaceCompare > 0)
  45. return false;
  46. return strcmp(p1.second, p2.second) < 0;
  47. }
  48. };
  49. struct Il2CppNameToTypeHandleHashTable : public Il2CppHashMap<std::pair<const char*, const char*>, Il2CppMetadataTypeHandle, NamespaceAndNamePairHash, NamespaceAndNamePairEquals>
  50. {
  51. typedef Il2CppHashMap<std::pair<const char*, const char*>, Il2CppMetadataTypeHandle, NamespaceAndNamePairHash, NamespaceAndNamePairEquals> Base;
  52. Il2CppNameToTypeHandleHashTable() : Base()
  53. {
  54. }
  55. };
  56. namespace il2cpp
  57. {
  58. namespace vm
  59. {
  60. const Il2CppAssembly* Image::GetAssembly(const Il2CppImage* image)
  61. {
  62. return image->assembly;
  63. }
  64. typedef il2cpp::vm::StackFrames::const_reverse_iterator StackReverseIterator;
  65. static bool IsSystemType(Il2CppClass* klass)
  66. {
  67. return strcmp(klass->namespaze, "System") == 0 && strcmp(klass->name, "Type") == 0;
  68. }
  69. static bool IsSystemReflectionAssembly(Il2CppClass* klass)
  70. {
  71. return strcmp(klass->namespaze, "System.Reflection") == 0 && strcmp(klass->name, "Assembly") == 0;
  72. }
  73. static StackReverseIterator GetNextImageFromStack(StackReverseIterator first, StackReverseIterator last)
  74. {
  75. for (StackReverseIterator it = first; it != last; it++)
  76. {
  77. Il2CppClass* klass = it->method->klass;
  78. if (klass->image != NULL && !IsSystemType(klass) && !IsSystemReflectionAssembly(klass))
  79. {
  80. return it;
  81. }
  82. }
  83. return last;
  84. }
  85. const Il2CppImage* Image::GetExecutingImage()
  86. {
  87. const il2cpp::vm::StackFrames& stack = *StackTrace::GetStackFrames();
  88. StackReverseIterator imageIt = GetNextImageFromStack(stack.rbegin(), stack.rend());
  89. if (imageIt != stack.rend())
  90. {
  91. return imageIt->method->klass->image;
  92. }
  93. // Fallback to corlib if no image is found
  94. return const_cast<Il2CppImage*>(Image::GetCorlib());
  95. }
  96. const Il2CppImage* Image::GetCallingImage()
  97. {
  98. const il2cpp::vm::StackFrames& stack = *StackTrace::GetStackFrames();
  99. StackReverseIterator imageIt = GetNextImageFromStack(stack.rbegin(), stack.rend());
  100. if (imageIt != stack.rend())
  101. {
  102. imageIt = GetNextImageFromStack(++imageIt, stack.rend());
  103. if (imageIt != stack.rend())
  104. {
  105. return imageIt->method->klass->image;
  106. }
  107. }
  108. // Fallback to corlib if no image is found
  109. return const_cast<Il2CppImage*>(Image::GetCorlib());
  110. }
  111. const char * Image::GetName(const Il2CppImage* image)
  112. {
  113. return image->name;
  114. }
  115. const char * Image::GetFileName(const Il2CppImage* image)
  116. {
  117. return image->name;
  118. }
  119. const MethodInfo* Image::GetEntryPoint(const Il2CppImage* image)
  120. {
  121. return MetadataCache::GetAssemblyEntryPoint(image);
  122. }
  123. Il2CppImage* Image::GetCorlib()
  124. {
  125. return il2cpp_defaults.corlib;
  126. }
  127. static baselib::ReentrantLock s_ClassFromNameMutex;
  128. static void AddNestedTypesToNametoClassHashTable(Il2CppNameToTypeHandleHashTable* hashTable, const char* namespaze, const std::string& parentName, Il2CppMetadataTypeHandle handle)
  129. {
  130. std::pair<const char*, const char*> namespaceAndName = MetadataCache::GetTypeNamespaceAndName(handle);
  131. std::string name = parentName + "/" + namespaceAndName.second;
  132. char* pName = (char*)IL2CPP_CALLOC(name.size() + 1, sizeof(char));
  133. strcpy(pName, name.c_str());
  134. hashTable->insert(std::make_pair(std::make_pair(namespaze, (const char*)pName), handle));
  135. void* iter = NULL;
  136. while (Il2CppMetadataTypeHandle nestedClass = MetadataCache::GetNestedTypes(handle, &iter))
  137. AddNestedTypesToNametoClassHashTable(hashTable, namespaze, name, nestedClass);
  138. }
  139. static void AddNestedTypesToNametoClassHashTable(const Il2CppImage* image, Il2CppMetadataTypeHandle handle)
  140. {
  141. std::pair<const char*, const char*> namespaceAndName = MetadataCache::GetTypeNamespaceAndName(handle);
  142. void* iter = NULL;
  143. while (Il2CppMetadataTypeHandle nestedClass = MetadataCache::GetNestedTypes(handle, &iter))
  144. {
  145. AddNestedTypesToNametoClassHashTable(image->nameToClassHashTable, namespaceAndName.first, namespaceAndName.second, nestedClass);
  146. }
  147. }
  148. // This must be called when the s_ClassFromNameMutex is held.
  149. static void AddTypeToNametoClassHashTable(const Il2CppImage* image, Il2CppMetadataTypeHandle typeHandle)
  150. {
  151. if (typeHandle == NULL)
  152. return;
  153. // don't add nested types
  154. if (MetadataCache::TypeIsNested(typeHandle))
  155. return;
  156. if (image != il2cpp_defaults.corlib)
  157. AddNestedTypesToNametoClassHashTable(image, typeHandle);
  158. image->nameToClassHashTable->insert(std::make_pair(MetadataCache::GetTypeNamespaceAndName(typeHandle), typeHandle));
  159. }
  160. void Image::InitNestedTypes(const Il2CppImage* image)
  161. {
  162. for (uint32_t index = 0; index < image->typeCount; index++)
  163. {
  164. Il2CppMetadataTypeHandle handle = MetadataCache::GetAssemblyTypeHandle(image, index);
  165. // don't add nested types
  166. if (MetadataCache::TypeIsNested(handle))
  167. return;
  168. AddNestedTypesToNametoClassHashTable(image, handle);
  169. }
  170. for (uint32_t index = 0; index < image->exportedTypeCount; index++)
  171. {
  172. Il2CppMetadataTypeHandle handle = MetadataCache::GetAssemblyExportedTypeHandle(image, index);
  173. // don't add nested types
  174. if (MetadataCache::TypeIsNested(handle))
  175. return;
  176. AddNestedTypesToNametoClassHashTable(image, handle);
  177. }
  178. }
  179. static void InitImageNameToTypeHandleHashTable(const Il2CppImage* image)
  180. {
  181. os::FastAutoLock lock(&s_ClassFromNameMutex);
  182. if (!image->nameToClassHashTable)
  183. {
  184. image->nameToClassHashTable = new Il2CppNameToTypeHandleHashTable();
  185. for (uint32_t index = 0; index < image->typeCount; index++)
  186. {
  187. AddTypeToNametoClassHashTable(image, MetadataCache::GetAssemblyTypeHandle(image, index));
  188. }
  189. for (uint32_t index = 0; index < image->exportedTypeCount; index++)
  190. {
  191. AddTypeToNametoClassHashTable(image, MetadataCache::GetAssemblyExportedTypeHandle(image, index));
  192. }
  193. }
  194. }
  195. Il2CppClass* Image::ClassFromName(const Il2CppImage* image, const char* namespaze, const char* name)
  196. {
  197. if (!image->nameToClassHashTable)
  198. {
  199. InitImageNameToTypeHandleHashTable(image);
  200. }
  201. Il2CppNameToTypeHandleHashTable::const_iterator iter = image->nameToClassHashTable->find(std::make_pair(namespaze, name));
  202. if (iter != image->nameToClassHashTable->end())
  203. return MetadataCache::GetTypeInfoFromHandle(iter->second);
  204. return NULL;
  205. }
  206. Il2CppMetadataTypeHandle Image::TypeHandleFromName(const Il2CppImage* image, const char* namespaze, const char* name)
  207. {
  208. if (!image->nameToClassHashTable)
  209. {
  210. InitImageNameToTypeHandleHashTable(image);
  211. }
  212. Il2CppNameToTypeHandleHashTable::const_iterator iter = image->nameToClassHashTable->find(std::make_pair(namespaze, name));
  213. if (iter != image->nameToClassHashTable->end())
  214. return iter->second;
  215. return NULL;
  216. }
  217. static bool IsExported(const Il2CppClass* type)
  218. {
  219. if ((type->flags & TYPE_ATTRIBUTE_VISIBILITY_MASK) == TYPE_ATTRIBUTE_PUBLIC)
  220. {
  221. return true;
  222. }
  223. if ((type->flags & TYPE_ATTRIBUTE_VISIBILITY_MASK) == TYPE_ATTRIBUTE_NESTED_PUBLIC)
  224. {
  225. IL2CPP_ASSERT(type->declaringType);
  226. return IsExported(type->declaringType);
  227. }
  228. return false;
  229. }
  230. void Image::GetTypes(const Il2CppImage* image, bool exportedOnly, TypeVector* target)
  231. {
  232. uint32_t typeCount = Image::GetNumTypes(image);
  233. target->reserve(typeCount);
  234. for (uint32_t sourceIndex = 0; sourceIndex < typeCount; sourceIndex++)
  235. {
  236. const Il2CppClass* type = Image::GetType(image, sourceIndex);
  237. if (strcmp(type->name, "<Module>") == 0)
  238. {
  239. continue;
  240. }
  241. if (exportedOnly && !IsExported(type))
  242. {
  243. continue;
  244. }
  245. target->push_back(type);
  246. }
  247. }
  248. Il2CppArray* Image::GetTypes(const Il2CppImage* image, bool exportedOnly)
  249. {
  250. TypeVector types;
  251. GetTypes(image, exportedOnly, &types);
  252. Il2CppArray* result = Array::New(il2cpp_defaults.monotype_class, (il2cpp_array_size_t)types.size());
  253. size_t index = 0;
  254. for (vm::TypeVector::const_iterator type = types.begin(); type != types.end(); ++type)
  255. {
  256. Il2CppReflectionType* reflectionType = Reflection::GetTypeObject(&(*type)->byval_arg);
  257. il2cpp_array_setref(result, index, reflectionType);
  258. index++;
  259. }
  260. return result;
  261. }
  262. uint32_t Image::GetNumTypes(const Il2CppImage* image)
  263. {
  264. return image->typeCount;
  265. }
  266. const Il2CppClass* Image::GetType(const Il2CppImage* image, AssemblyTypeIndex index)
  267. {
  268. IL2CPP_ASSERT(index <= std::numeric_limits<AssemblyTypeIndex>::max());
  269. return MetadataCache::GetTypeInfoFromHandle(MetadataCache::GetAssemblyTypeHandle(image, index));
  270. }
  271. static bool StringsMatch(const char* left, const char* right, bool ignoreCase)
  272. {
  273. if (!ignoreCase)
  274. {
  275. return strcmp(left, right) == 0;
  276. }
  277. else
  278. {
  279. utils::VmStringUtils::CaseInsensitiveComparer comparer;
  280. return comparer(left, right);
  281. }
  282. }
  283. static bool ClassMatches(Il2CppMetadataTypeHandle typeHandle, const char* namespaze, bool ignoreCase, const char* name)
  284. {
  285. if (MetadataCache::TypeIsNested(typeHandle))
  286. return false;
  287. std::pair<const char*, const char*> namespaceAndName = MetadataCache::GetTypeNamespaceAndName(typeHandle);
  288. return StringsMatch(name, namespaceAndName.second, ignoreCase) && StringsMatch(namespaze, namespaceAndName.first, ignoreCase);
  289. }
  290. static Il2CppClass* FindClassMatching(const Il2CppImage* image, const char* namespaze, const char *name, bool ignoreCase)
  291. {
  292. for (uint32_t i = 0; i < image->typeCount; i++)
  293. {
  294. Il2CppMetadataTypeHandle typeHandle = MetadataCache::GetAssemblyTypeHandle(image, i);
  295. if (ClassMatches(typeHandle, namespaze, ignoreCase, name))
  296. return MetadataCache::GetTypeInfoFromHandle(typeHandle);
  297. }
  298. return NULL;
  299. }
  300. static Il2CppClass* FindExportedClassMatching(const Il2CppImage* image, const char* namespaze, const char *name, bool ignoreCase)
  301. {
  302. for (uint32_t i = 0; i < image->exportedTypeCount; i++)
  303. {
  304. Il2CppMetadataTypeHandle typeHandle = MetadataCache::GetAssemblyExportedTypeHandle(image, i);
  305. if (ClassMatches(typeHandle, namespaze, ignoreCase, name))
  306. return MetadataCache::GetTypeInfoFromHandle(typeHandle);
  307. }
  308. return NULL;
  309. }
  310. static Il2CppClass* FindNestedType(Il2CppClass* klass, const char* name, bool ignoreCase)
  311. {
  312. void* iter = NULL;
  313. while (Il2CppClass* nestedType = Class::GetNestedTypes(klass, &iter))
  314. {
  315. if (StringsMatch(name, nestedType->name, ignoreCase))
  316. return nestedType;
  317. }
  318. return NULL;
  319. }
  320. Il2CppClass* Image::FromTypeNameParseInfo(const Il2CppImage* image, const TypeNameParseInfo &info, bool ignoreCase)
  321. {
  322. const char* ns = info.ns().c_str();
  323. const char* name = info.name().c_str();
  324. Il2CppClass *parent_class = FindClassMatching(image, ns, name, ignoreCase);
  325. if (parent_class == NULL)
  326. {
  327. parent_class = FindExportedClassMatching(image, ns, name, ignoreCase);
  328. if (parent_class == NULL)
  329. return NULL;
  330. }
  331. std::vector<std::string>::const_iterator it = info.nested().begin();
  332. while (it != info.nested().end())
  333. {
  334. parent_class = FindNestedType(parent_class, (*it).c_str(), ignoreCase);
  335. if (parent_class == NULL)
  336. return NULL;
  337. ++it;
  338. }
  339. return parent_class;
  340. }
  341. static baselib::ReentrantLock s_Mutex;
  342. static std::vector<Image::EmbeddedResourceData> s_CachedResourceData;
  343. static std::map<Il2CppReflectionAssembly*, void*> s_CachedMemoryMappedResourceFiles;
  344. void Image::CacheMemoryMappedResourceFile(Il2CppReflectionAssembly* assembly, void* memoryMappedFile)
  345. {
  346. os::FastAutoLock lock(&s_Mutex);
  347. s_CachedMemoryMappedResourceFiles[assembly] = memoryMappedFile;
  348. }
  349. void* Image::GetCachedMemoryMappedResourceFile(Il2CppReflectionAssembly* assembly)
  350. {
  351. os::FastAutoLock lock(&s_Mutex);
  352. std::map<Il2CppReflectionAssembly*, void*>::iterator entry = s_CachedMemoryMappedResourceFiles.find(assembly);
  353. if (entry != s_CachedMemoryMappedResourceFiles.end())
  354. return entry->second;
  355. return NULL;
  356. }
  357. void Image::CacheResourceData(EmbeddedResourceRecord record, void* data)
  358. {
  359. os::FastAutoLock lock(&s_Mutex);
  360. s_CachedResourceData.push_back(EmbeddedResourceData(record, data));
  361. }
  362. void* Image::GetCachedResourceData(const Il2CppImage* image, const std::string& name)
  363. {
  364. os::FastAutoLock lock(&s_Mutex);
  365. for (std::vector<EmbeddedResourceData>::iterator it = s_CachedResourceData.begin(); it != s_CachedResourceData.end(); ++it)
  366. {
  367. if (it->record.image == image && it->record.name == name)
  368. return it->data;
  369. }
  370. return NULL;
  371. }
  372. void Image::ClearCachedResourceData()
  373. {
  374. os::FastAutoLock lock(&s_Mutex);
  375. for (std::map<Il2CppReflectionAssembly*, void*>::iterator i = s_CachedMemoryMappedResourceFiles.begin(); i != s_CachedMemoryMappedResourceFiles.end(); ++i)
  376. utils::MemoryMappedFile::Unmap(i->second);
  377. s_CachedMemoryMappedResourceFiles.clear();
  378. s_CachedResourceData.clear();
  379. }
  380. } /* namespace vm */
  381. } /* namespace il2cpp */