AsyncLoader.cpp 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. // Copyright (C) 2009-2015, Panagiotis Christopoulos Charitos.
  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/resource/AsyncLoader.h"
  7. #include "anki/util/HighRezTimer.h"
  8. #include "anki/util/Atomic.h"
  9. #include "anki/util/Functions.h"
  10. namespace anki {
  11. //==============================================================================
  12. class Task: public AsyncLoader::Task
  13. {
  14. public:
  15. F32 m_sleepTime = 0.0;
  16. Barrier* m_barrier = nullptr;
  17. Atomic<U32>* m_count = nullptr;
  18. I32 m_id = -1;
  19. Task(F32 time, Barrier* barrier, Atomic<U32>* count, I32 id = -1)
  20. : m_sleepTime(time),
  21. m_barrier(barrier),
  22. m_count(count),
  23. m_id(id)
  24. {}
  25. Error operator()()
  26. {
  27. if(m_count)
  28. {
  29. auto x = m_count->fetchAdd(1);
  30. if(m_id >= 0)
  31. {
  32. if(m_id != static_cast<I32>(x))
  33. {
  34. ANKI_LOGE("Wrong excecution order");
  35. return ErrorCode::FUNCTION_FAILED;
  36. }
  37. }
  38. return ErrorCode::NONE;
  39. }
  40. if(m_sleepTime != 0.0)
  41. {
  42. HighRezTimer::sleep(m_sleepTime);
  43. }
  44. if(m_barrier)
  45. {
  46. m_barrier->wait();
  47. }
  48. return ErrorCode::NONE;
  49. }
  50. };
  51. //==============================================================================
  52. class MemTask: public AsyncLoader::Task
  53. {
  54. public:
  55. HeapAllocator<U8> m_alloc;
  56. Barrier* m_barrier = nullptr;
  57. MemTask(HeapAllocator<U8> alloc, Barrier* barrier)
  58. : m_alloc(alloc),
  59. m_barrier(barrier)
  60. {}
  61. Error operator()()
  62. {
  63. void* mem = m_alloc.allocate(10);
  64. if(!mem) return ErrorCode::FUNCTION_FAILED;
  65. HighRezTimer::sleep(0.1);
  66. m_alloc.deallocate(mem, 10);
  67. if(m_barrier)
  68. {
  69. m_barrier->wait();
  70. }
  71. return ErrorCode::NONE;
  72. }
  73. };
  74. //==============================================================================
  75. ANKI_TEST(Resource, AsyncLoader)
  76. {
  77. HeapAllocator<U8> alloc(allocAligned, nullptr);
  78. // Simple create destroy
  79. {
  80. AsyncLoader a;
  81. ANKI_TEST_EXPECT_NO_ERR(a.create(alloc));
  82. }
  83. // Simple task that will finish
  84. {
  85. AsyncLoader a;
  86. ANKI_TEST_EXPECT_NO_ERR(a.create(alloc));
  87. Barrier barrier(2);
  88. ANKI_TEST_EXPECT_NO_ERR(a.newTask<Task>(0.0, &barrier, nullptr));
  89. barrier.wait();
  90. }
  91. // Many tasks that will finish
  92. {
  93. AsyncLoader a;
  94. ANKI_TEST_EXPECT_NO_ERR(a.create(alloc));
  95. Barrier barrier(2);
  96. Atomic<U32> counter = {0};
  97. for(U i = 0; i < 100; i++)
  98. {
  99. Barrier* pbarrier = nullptr;
  100. if(i == 99)
  101. {
  102. pbarrier = &barrier;
  103. }
  104. ANKI_TEST_EXPECT_NO_ERR(a.newTask<Task>(0.0, pbarrier, &counter));
  105. }
  106. barrier.wait();
  107. ANKI_TEST_EXPECT_EQ(counter.load(), 100);
  108. }
  109. // Many tasks that will _not_ finish
  110. {
  111. AsyncLoader a;
  112. ANKI_TEST_EXPECT_NO_ERR(a.create(alloc));
  113. for(U i = 0; i < 100; i++)
  114. {
  115. ANKI_TEST_EXPECT_NO_ERR(a.newTask<Task>(0.0, nullptr, nullptr));
  116. }
  117. }
  118. // Tasks that allocate
  119. {
  120. AsyncLoader a;
  121. ANKI_TEST_EXPECT_NO_ERR(a.create(alloc));
  122. Barrier barrier(2);
  123. for(U i = 0; i < 10; i++)
  124. {
  125. Barrier* pbarrier = nullptr;
  126. if(i == 9)
  127. {
  128. pbarrier = &barrier;
  129. }
  130. ANKI_TEST_EXPECT_NO_ERR(a.newTask<MemTask>(alloc, pbarrier));
  131. }
  132. barrier.wait();
  133. }
  134. // Tasks that allocate and never finished
  135. {
  136. AsyncLoader a;
  137. ANKI_TEST_EXPECT_NO_ERR(a.create(alloc));
  138. for(U i = 0; i < 10; i++)
  139. {
  140. ANKI_TEST_EXPECT_NO_ERR(a.newTask<MemTask>(alloc, nullptr));
  141. }
  142. }
  143. // Random struf
  144. {
  145. AsyncLoader a;
  146. ANKI_TEST_EXPECT_NO_ERR(a.create(alloc));
  147. Barrier barrier(2);
  148. Atomic<U32> counter = {0};
  149. for(U i = 0; i < 10; i++)
  150. {
  151. Barrier* pbarrier = nullptr;
  152. if(i == 9)
  153. {
  154. pbarrier = &barrier;
  155. }
  156. ANKI_TEST_EXPECT_NO_ERR(
  157. a.newTask<Task>(randRange(0.0, 0.5), pbarrier, &counter, i));
  158. }
  159. barrier.wait();
  160. ANKI_TEST_EXPECT_EQ(counter.load(), 10);
  161. }
  162. }
  163. } // end namespace anki