226 lines
6.7 KiB

  1. //
  2. // Lol Engine
  3. //
  4. // Copyright © 2010—2015 Sam Hocevar <sam@hocevar.net>
  5. // © 2014—2015 Benjamin "Touky" Huet <huet.benjamin@gmail.com>
  6. //
  7. // This program is free software. It comes without any warranty, to
  8. // the extent permitted by applicable law. You can redistribute it
  9. // and/or modify it under the terms of the Do What the Fuck You Want
  10. // to Public License, Version 2, as published by the WTFPL Task Force.
  11. // See http://www.wtfpl.net/ for more details.
  12. //
  13. #include <lol/engine-internal.h>
  14. #include <lolunit.h>
  15. namespace lol
  16. {
  17. lolunit_declare_fixture(thread_test)
  18. {
  19. //FileUpdateTesterJob ---------------------------------------------------------
  20. class UnitTestJob : public ThreadJob
  21. {
  22. friend class UnitTestThreadManager;
  23. public:
  24. char const *GetName() { return "<UnitTestJob>"; }
  25. UnitTestJob() : ThreadJob(ThreadJobType::WORK_TODO)
  26. { }
  27. bool IsDone()
  28. {
  29. return m_done;
  30. }
  31. protected:
  32. virtual bool DoWork()
  33. {
  34. Timer timer;
  35. m_done = false;
  36. Log::Info(Line("%s: STARTED WORK"), GetName());
  37. timer.Wait(2.f);
  38. Log::Info(Line("%s: ENDED WORK"), GetName());
  39. m_done = true;
  40. return true;
  41. }
  42. bool m_done = false;
  43. };
  44. //Unit test thread manager
  45. class UnitTestThreadManager : public BaseThreadManager
  46. {
  47. typedef BaseThreadManager super;
  48. struct UnitTestStatusBase : public StructSafeEnum
  49. {
  50. enum Type
  51. {
  52. NOT_QUEUED,
  53. QUEUED,
  54. RETRIEVED,
  55. DONE,
  56. };
  57. protected:
  58. virtual bool BuildEnumMap(map<int64_t, String>& enum_map)
  59. {
  60. enum_map[NOT_QUEUED] = "NOT_QUEUED";
  61. enum_map[QUEUED] = "QUEUED";
  62. enum_map[RETRIEVED] = "RETRIEVED";
  63. enum_map[DONE] = "DONE";
  64. return true;
  65. }
  66. };
  67. typedef SafeEnum<UnitTestStatusBase> UnitTestStatus;
  68. public:
  69. char const *GetName() { return "<UnitTestThreadManager>"; }
  70. UnitTestThreadManager() : BaseThreadManager(4, 1)
  71. { }
  72. virtual ~UnitTestThreadManager()
  73. { }
  74. void AddJob(ThreadJob* job)
  75. {
  76. Log::Info(Line("%s DISPATCHING JOB %s"), GetName(), job->GetName());
  77. DispatchJob(job);
  78. }
  79. bool GetWorkResult(array<ThreadJob*>& results)
  80. {
  81. results += m_job_result;
  82. m_job_result.Empty();
  83. Log::Info(Line("%s GETWORKRESULT (%i)"), GetName(), results.count());
  84. return results.Count() > 0;
  85. }
  86. virtual void TickGame(float seconds)
  87. {
  88. switch (m_status.ToScalar())
  89. {
  90. case UnitTestStatus::NOT_QUEUED:
  91. if (!!GetDispatchCount())
  92. {
  93. Log::Info(Line("%s TICKGAME %s"), GetName(), m_status.ToString().C());
  94. m_status = UnitTestStatus::QUEUED;
  95. }
  96. break;
  97. case UnitTestStatus::QUEUED:
  98. #if !defined(LOL_FEATURE_THREADS) || !LOL_FEATURE_THREADS
  99. if (!GetDispatchedCount())
  100. #else //LOL_FEATURE_THREADS
  101. if (GetDispatchedCount())
  102. #endif
  103. {
  104. Log::Info(Line("%s TICKGAME %s"), GetName(), m_status.ToString().C());
  105. m_status = UnitTestStatus::RETRIEVED;
  106. }
  107. break;
  108. case UnitTestStatus::RETRIEVED:
  109. if (m_job_result.count() == 4)
  110. {
  111. Log::Info(Line("%s TICKGAME %s"), GetName(), m_status.ToString().C());
  112. m_status = UnitTestStatus::DONE;
  113. }
  114. break;
  115. default:
  116. break;
  117. }
  118. //Debug error fix-up
  119. #if !LOL_BUILD_RELEASE
  120. m_tickstate = STATE_PRETICK_GAME;
  121. #endif
  122. super::TickGame(seconds);
  123. }
  124. bool IsDone() { return m_status == UnitTestStatus::DONE; }
  125. int Test_GetDispatchCount() { return GetDispatchCount(); }
  126. int Test_GetDispatchedCount() { return GetDispatchedCount(); }
  127. protected:
  128. virtual void TreatResult(ThreadJob* result) { m_job_result << result; }
  129. array<ThreadJob*> m_job_result;
  130. UnitTestStatus m_status;
  131. };
  132. UnitTestThreadManager m_manager;
  133. void SetUp()
  134. {
  135. }
  136. void TearDown()
  137. {
  138. }
  139. lolunit_declare_test(threads)
  140. {
  141. Log::Info(Line("%s START"), m_manager.GetName());
  142. //Start threads manager
  143. m_manager.Start();
  144. Log::Info(Line("%s STARTED"), m_manager.GetName());
  145. UnitTestJob job[4];
  146. lolunit_assert_equal(0, m_manager.Test_GetDispatchCount());
  147. m_manager.AddJob(&job[0]);
  148. lolunit_assert_equal(1, m_manager.Test_GetDispatchCount());
  149. lolunit_assert_equal(false, job[0].IsDone());
  150. bool dispatch_check = true;
  151. while (!m_manager.IsDone())
  152. {
  153. m_manager.TickGame(1.f / 60.f);
  154. if (dispatch_check)
  155. {
  156. lolunit_assert_equal(0, m_manager.Test_GetDispatchCount());
  157. #if !defined(LOL_FEATURE_THREADS) || !LOL_FEATURE_THREADS
  158. lolunit_assert_equal(0, m_manager.Test_GetDispatchedCount());
  159. #else //LOL_FEATURE_THREADS
  160. lolunit_assert_equal(1, m_manager.Test_GetDispatchedCount());
  161. #endif
  162. m_manager.AddJob(&job[1]);
  163. m_manager.AddJob(&job[2]);
  164. m_manager.AddJob(&job[3]);
  165. dispatch_check = false;
  166. }
  167. }
  168. lolunit_assert_equal(true, job[0].IsDone());
  169. lolunit_assert_equal(true, job[1].IsDone());
  170. lolunit_assert_equal(true, job[2].IsDone());
  171. lolunit_assert_equal(true, job[3].IsDone());
  172. array<ThreadJob*> results;
  173. m_manager.GetWorkResult(results);
  174. lolunit_assert_equal(4, results.count());
  175. Log::Info(Line("%s STOP"), m_manager.GetName());
  176. //Stop manager
  177. m_manager.Stop();
  178. Log::Info(Line("%s STOPPED"), m_manager.GetName());
  179. }
  180. lolunit_declare_test(queue_try_push)
  181. {
  182. queue<int, 1> q;
  183. bool b1 = q.try_push(0);
  184. lolunit_assert_equal(true, b1);
  185. bool b2 = q.try_push(1);
  186. lolunit_assert_equal(false, b2);
  187. }
  188. lolunit_declare_test(queue_try_pop)
  189. {
  190. queue<int, 1> q;
  191. int tmp;
  192. q.push(42);
  193. bool b1 = q.try_pop(tmp);
  194. lolunit_assert_equal(true, b1);
  195. lolunit_assert_equal(42, tmp);
  196. bool b2 = q.try_pop(tmp);
  197. lolunit_assert_equal(false, b2);
  198. lolunit_assert_equal(42, tmp);
  199. }
  200. };
  201. } /* namespace lol */