SparseArray.cpp 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446
  1. // Copyright (C) 2009-2021, Panagiotis Christopoulos Charitos and contributors.
  2. // All rights reserved.
  3. // Code licensed under the BSD License.
  4. // http://www.anki3d.org/LICENSE
  5. #include <Tests/Framework/Framework.h>
  6. #include <anki/util/SparseArray.h>
  7. #include <anki/util/HighRezTimer.h>
  8. #include <unordered_map>
  9. #include <ctime>
  10. #include <algorithm>
  11. #include <malloc.h>
  12. namespace anki
  13. {
  14. namespace
  15. {
  16. static I64 constructor0Count = 0;
  17. static I64 constructor1Count = 0;
  18. static I64 constructor2Count = 0;
  19. static I64 constructor3Count = 0;
  20. static I64 destructorCount = 0;
  21. static I64 copyCount = 0;
  22. static I64 moveCount = 0;
  23. class SAFoo
  24. {
  25. public:
  26. int m_x;
  27. SAFoo()
  28. : m_x(0)
  29. {
  30. ++constructor0Count;
  31. }
  32. SAFoo(int x)
  33. : m_x(x)
  34. {
  35. ++constructor1Count;
  36. }
  37. SAFoo(const SAFoo& b)
  38. : m_x(b.m_x)
  39. {
  40. ++constructor2Count;
  41. }
  42. SAFoo(SAFoo&& b)
  43. : m_x(b.m_x)
  44. {
  45. b.m_x = 0;
  46. ++constructor3Count;
  47. }
  48. ~SAFoo()
  49. {
  50. ++destructorCount;
  51. }
  52. SAFoo& operator=(const SAFoo& b)
  53. {
  54. m_x = b.m_x;
  55. ++copyCount;
  56. return *this;
  57. }
  58. SAFoo& operator=(SAFoo&& b)
  59. {
  60. m_x = b.m_x;
  61. b.m_x = 0;
  62. ++moveCount;
  63. return *this;
  64. }
  65. static void checkCalls()
  66. {
  67. ANKI_TEST_EXPECT_EQ(constructor0Count, 0); // default
  68. ANKI_TEST_EXPECT_GT(constructor1Count, 0);
  69. ANKI_TEST_EXPECT_EQ(constructor2Count, 0); // copy
  70. ANKI_TEST_EXPECT_GT(constructor3Count, 0); // move
  71. ANKI_TEST_EXPECT_EQ(destructorCount, constructor1Count + constructor3Count);
  72. ANKI_TEST_EXPECT_EQ(copyCount, 0); // copy
  73. ANKI_TEST_EXPECT_GEQ(moveCount, 0); // move
  74. constructor0Count = constructor1Count = constructor2Count = constructor3Count = destructorCount = copyCount =
  75. moveCount = 0;
  76. }
  77. };
  78. } // namespace
  79. } // namespace anki
  80. ANKI_TEST(Util, SparseArray)
  81. {
  82. HeapAllocator<U8> alloc(allocAligned, nullptr);
  83. // Set same key
  84. {
  85. SparseArray<PtrSize> arr;
  86. arr.emplace(alloc, 1000, 123);
  87. arr.emplace(alloc, 1000, 124);
  88. auto it = arr.find(1000);
  89. ANKI_TEST_EXPECT_EQ(*it, 124);
  90. arr.erase(alloc, it);
  91. }
  92. // Check destroy and grow
  93. {
  94. SparseArray<SAFoo> arr(64, 2);
  95. arr.emplace(alloc, 64 * 1, 123);
  96. arr.emplace(alloc, 64 * 2, 124);
  97. arr.emplace(alloc, 64 * 3, 125);
  98. ANKI_TEST_EXPECT_EQ(arr.find(64 * 1)->m_x, 123);
  99. ANKI_TEST_EXPECT_EQ(arr.find(64 * 2)->m_x, 124);
  100. ANKI_TEST_EXPECT_EQ(arr.find(64 * 3)->m_x, 125);
  101. arr.destroy(alloc);
  102. SAFoo::checkCalls();
  103. }
  104. // Do complex insertions
  105. {
  106. SparseArray<SAFoo, U32> arr(64, 3);
  107. arr.emplace(alloc, 64 * 0 - 1, 1);
  108. // Linear probing to 0
  109. arr.emplace(alloc, 64 * 1 - 1, 2);
  110. // Linear probing to 1
  111. arr.emplace(alloc, 64 * 2 - 1, 3);
  112. // Linear probing to 2
  113. arr.emplace(alloc, 1, 3);
  114. // Swap
  115. arr.emplace(alloc, 64 * 1, 3);
  116. ANKI_TEST_EXPECT_EQ(arr.getSize(), 5);
  117. arr.destroy(alloc);
  118. SAFoo::checkCalls();
  119. }
  120. // Fuzzy test
  121. {
  122. const U MAX = 10000;
  123. SparseArray<SAFoo, U32> arr;
  124. std::vector<int> numbers;
  125. srand(U32(time(nullptr)));
  126. // Insert random
  127. for(U i = 0; i < MAX; ++i)
  128. {
  129. I32 num;
  130. while(1)
  131. {
  132. num = rand();
  133. if(std::find(numbers.begin(), numbers.end(), num) == numbers.end())
  134. {
  135. // Not found
  136. ANKI_TEST_EXPECT_EQ(arr.find(num), arr.getEnd());
  137. arr.emplace(alloc, num, num);
  138. ANKI_TEST_EXPECT_EQ(arr.getSize(), i + 1);
  139. numbers.push_back(num);
  140. break;
  141. }
  142. else
  143. {
  144. // Found
  145. ANKI_TEST_EXPECT_NEQ(arr.find(num), arr.getEnd());
  146. }
  147. }
  148. arr.validate();
  149. }
  150. ANKI_TEST_EXPECT_EQ(arr.getSize(), MAX);
  151. // Remove randomly
  152. U count = MAX;
  153. while(count--)
  154. {
  155. U idx = rand() % (count + 1);
  156. int num = numbers[idx];
  157. numbers.erase(numbers.begin() + idx);
  158. auto it = arr.find(num);
  159. ANKI_TEST_EXPECT_NEQ(it, arr.getEnd());
  160. ANKI_TEST_EXPECT_EQ(it->m_x, num);
  161. arr.erase(alloc, it);
  162. arr.validate();
  163. }
  164. }
  165. // Fuzzy test #2: Do random insertions and removals
  166. {
  167. const U MAX = 10000;
  168. SparseArray<SAFoo, U64> arr;
  169. using StlMap =
  170. std::unordered_map<int, int, std::hash<int>, std::equal_to<int>, HeapAllocator<std::pair<const int, int>>>;
  171. StlMap map(10, std::hash<int>(), std::equal_to<int>(), alloc);
  172. for(U i = 0; i < MAX; ++i)
  173. {
  174. const Bool insert = (rand() & 1) || arr.getSize() == 0;
  175. if(insert)
  176. {
  177. const I32 idx = rand();
  178. if(map.find(idx) != map.end())
  179. {
  180. continue;
  181. }
  182. arr.emplace(alloc, idx, idx + 1);
  183. map[idx] = idx + 1;
  184. arr.validate();
  185. }
  186. else
  187. {
  188. const U idx = U(rand()) % map.size();
  189. auto it = std::next(std::begin(map), idx);
  190. int key = it->first;
  191. auto it2 = arr.find(key);
  192. ANKI_TEST_EXPECT_NEQ(it2, arr.getEnd());
  193. ANKI_TEST_EXPECT_EQ(it->second, it2->m_x);
  194. map.erase(it);
  195. arr.erase(alloc, it2);
  196. ANKI_TEST_EXPECT_EQ(arr.find(key), arr.getEnd());
  197. arr.validate();
  198. }
  199. // Iterate and check
  200. {
  201. StlMap bMap = map;
  202. auto it = arr.getBegin();
  203. while(it != arr.getEnd())
  204. {
  205. I32 key = it->m_x - 1;
  206. auto it2 = bMap.find(key);
  207. ANKI_TEST_EXPECT_NEQ(it2, bMap.end());
  208. bMap.erase(it2);
  209. ++it;
  210. }
  211. }
  212. }
  213. arr.destroy(alloc);
  214. // Check what the SparseArray have called
  215. SAFoo::checkCalls();
  216. }
  217. }
  218. static I64 akAllocSize = 0;
  219. static I64 akMaxAllocSize = 0;
  220. static ANKI_DONT_INLINE void* allocAlignedAk(void* userData, void* ptr, PtrSize size, PtrSize alignment)
  221. {
  222. if(ptr == nullptr)
  223. {
  224. #if ANKI_OS_LINUX
  225. akAllocSize += size;
  226. akMaxAllocSize = max(akMaxAllocSize, akAllocSize);
  227. #endif
  228. return malloc(size);
  229. }
  230. else
  231. {
  232. #if ANKI_OS_LINUX
  233. PtrSize s = malloc_usable_size(ptr);
  234. akAllocSize -= s;
  235. #endif
  236. free(ptr);
  237. return nullptr;
  238. }
  239. }
  240. static I64 stlAllocSize = 0;
  241. static I64 stlMaxAllocSize = 0;
  242. static ANKI_DONT_INLINE void* allocAlignedStl(void* userData, void* ptr, PtrSize size, PtrSize alignment)
  243. {
  244. if(ptr == nullptr)
  245. {
  246. #if ANKI_OS_LINUX
  247. stlAllocSize += size;
  248. stlMaxAllocSize = max(stlMaxAllocSize, stlAllocSize);
  249. #endif
  250. return malloc(size);
  251. }
  252. else
  253. {
  254. #if ANKI_OS_LINUX
  255. PtrSize s = malloc_usable_size(ptr);
  256. stlAllocSize -= s;
  257. #endif
  258. free(ptr);
  259. return nullptr;
  260. }
  261. }
  262. ANKI_TEST(Util, SparseArrayBench)
  263. {
  264. HeapAllocator<U8> allocAk(allocAlignedAk, nullptr);
  265. HeapAllocator<U8> allocStl(allocAlignedStl, nullptr);
  266. HeapAllocator<U8> allocTml(allocAligned, nullptr);
  267. using StlMap =
  268. std::unordered_map<int, int, std::hash<int>, std::equal_to<int>, HeapAllocator<std::pair<const int, int>>>;
  269. StlMap stdMap(10, std::hash<int>(), std::equal_to<int>(), allocStl);
  270. using AkMap = SparseArray<int, U32>;
  271. AkMap akMap(256, U32(log2(256.0f)), 0.90f);
  272. HighRezTimer timer;
  273. const U COUNT = 1024 * 1024 * 6;
  274. // Create a huge set
  275. std::vector<int> vals;
  276. {
  277. std::unordered_map<int, int> tmpMap;
  278. for(U i = 0; i < COUNT; ++i)
  279. {
  280. // Put unique keys
  281. int v;
  282. do
  283. {
  284. v = rand();
  285. } while(tmpMap.find(v) != tmpMap.end() && v != 0);
  286. tmpMap[v] = 1;
  287. vals.push_back(v);
  288. }
  289. }
  290. // Insertion
  291. {
  292. // AnkI
  293. timer.start();
  294. for(U i = 0; i < COUNT; ++i)
  295. {
  296. akMap.emplace(allocAk, vals[i], vals[i]);
  297. }
  298. timer.stop();
  299. Second akTime = timer.getElapsedTime();
  300. // STL
  301. timer.start();
  302. for(U i = 0; i < COUNT; ++i)
  303. {
  304. stdMap[vals[i]] = vals[i];
  305. }
  306. timer.stop();
  307. Second stlTime = timer.getElapsedTime();
  308. ANKI_TEST_LOGI("Inserting bench: STL %f AnKi %f | %f%%", stlTime, akTime, stlTime / akTime * 100.0);
  309. }
  310. // Search
  311. {
  312. // Search in random order
  313. std::random_shuffle(vals.begin(), vals.end());
  314. int count = 0;
  315. // Find values AnKi
  316. timer.start();
  317. for(U i = 0; i < COUNT; ++i)
  318. {
  319. auto it = akMap.find(vals[i]);
  320. count += *it;
  321. }
  322. timer.stop();
  323. Second akTime = timer.getElapsedTime();
  324. // Find values STL
  325. timer.start();
  326. for(U i = 0; i < COUNT; ++i)
  327. {
  328. count += stdMap[vals[i]];
  329. }
  330. timer.stop();
  331. Second stlTime = timer.getElapsedTime();
  332. // Print the "count" so that the compiler won't optimize it
  333. ANKI_TEST_LOGI("Find bench: STL %f AnKi %f | %f%% (r:%d)", stlTime, akTime, stlTime / akTime * 100.0, count);
  334. }
  335. // Mem usage
  336. const I64 stlMemUsage = stlMaxAllocSize + sizeof(stdMap);
  337. const I64 akMemUsage = akMaxAllocSize + sizeof(akMap);
  338. ANKI_TEST_LOGI("Max mem usage: STL %li AnKi %li | %f%% (At any given time what was the max mem usage)", stlMemUsage,
  339. akMemUsage, F64(stlMemUsage) / F32(akMemUsage) * 100.0f);
  340. // Deletes
  341. {
  342. // Remove in random order
  343. std::random_shuffle(vals.begin(), vals.end());
  344. // Random delete AnKi
  345. Second akTime = 0.0;
  346. for(U i = 0; i < vals.size(); ++i)
  347. {
  348. auto it = akMap.find(vals[i]);
  349. timer.start();
  350. akMap.erase(allocAk, it);
  351. timer.stop();
  352. akTime += timer.getElapsedTime();
  353. }
  354. // Random delete STL
  355. Second stlTime = 0.0;
  356. for(U i = 0; i < vals.size(); ++i)
  357. {
  358. auto it = stdMap.find(vals[i]);
  359. timer.start();
  360. stdMap.erase(it);
  361. timer.stop();
  362. stlTime += timer.getElapsedTime();
  363. }
  364. ANKI_TEST_LOGI("Deleting bench: STL %f AnKi %f | %f%%\n", stlTime, akTime, stlTime / akTime * 100.0);
  365. }
  366. akMap.destroy(allocAk);
  367. }