threadSafeRefCountTest.cpp 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) 2014 GarageGames, LLC
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to
  6. // deal in the Software without restriction, including without limitation the
  7. // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
  8. // sell copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
  20. // IN THE SOFTWARE.
  21. //-----------------------------------------------------------------------------
  22. #ifdef TORQUE_TESTS_ENABLED
  23. #include "testing/unitTesting.h"
  24. #include "platform/threads/threadSafeRefCount.h"
  25. #include "platform/threads/thread.h"
  26. #include "core/util/tVector.h"
  27. #include "console/console.h"
  28. FIXTURE(ThreadSafeRefCount)
  29. {
  30. public:
  31. struct TestObjectDtor : public ThreadSafeRefCount<TestObjectDtor>
  32. {
  33. bool &flag;
  34. TestObjectDtor(bool &f) : flag(f)
  35. {
  36. flag = false;
  37. }
  38. ~TestObjectDtor()
  39. {
  40. flag = true;
  41. }
  42. };
  43. typedef ThreadSafeRef<TestObjectDtor> TestObjectDtorRef;
  44. enum
  45. {
  46. NUM_ADD_REFS_PER_THREAD = 10,
  47. NUM_EXTRA_REFS_PER_THREAD = 10,
  48. NUM_THREADS = 10
  49. };
  50. class TestObject : public ThreadSafeRefCount<TestObject> {};
  51. typedef ThreadSafeRef<TestObject> TestObjectRef;
  52. class TestThread : public Thread
  53. {
  54. public:
  55. TestObjectRef mRef;
  56. Vector<TestObjectRef> mExtraRefs;
  57. TestThread(TestObjectRef ref) : mRef(ref) {}
  58. void run(void* arg)
  59. {
  60. if (!arg)
  61. {
  62. // Create references.
  63. for (U32 i = 0; i < NUM_ADD_REFS_PER_THREAD; i++)
  64. mRef->addRef();
  65. mExtraRefs.setSize(NUM_EXTRA_REFS_PER_THREAD);
  66. for (U32 i = 0; i < NUM_EXTRA_REFS_PER_THREAD; i++)
  67. mExtraRefs[i] = mRef;
  68. }
  69. else
  70. {
  71. // Clear references.
  72. mExtraRefs.clear();
  73. for (U32 i = 0; i < NUM_ADD_REFS_PER_THREAD; i++)
  74. mRef->release();
  75. }
  76. }
  77. };
  78. };
  79. TEST_FIX(ThreadSafeRefCount, Serial)
  80. {
  81. bool deleted = false;
  82. TestObjectDtorRef ref1 = new TestObjectDtor(deleted);
  83. ASSERT_FALSE(deleted);
  84. EXPECT_FALSE(ref1->isShared());
  85. EXPECT_TRUE(ref1 != NULL);
  86. TestObjectDtorRef ref2 = ref1;
  87. EXPECT_TRUE(ref1->isShared());
  88. EXPECT_TRUE(ref2->isShared());
  89. EXPECT_EQ(ref1, ref2);
  90. ref1 = NULL;
  91. EXPECT_FALSE(ref2->isShared());
  92. ref2 = NULL;
  93. ASSERT_TRUE(deleted);
  94. }
  95. TEST_FIX(ThreadSafeRefCount, Concurrent)
  96. {
  97. TestObjectRef mRef = new TestObject;
  98. EXPECT_EQ(2, mRef->getRefCount()); // increments of 2
  99. Vector<TestThread*> threads;
  100. threads.setSize(NUM_THREADS);
  101. // Create threads.
  102. for (U32 i = 0; i < NUM_THREADS; i++)
  103. threads[i] = new TestThread(mRef);
  104. // Run phase 1: create references.
  105. for (U32 i = 0; i < NUM_THREADS; i++)
  106. threads[i]->start(NULL);
  107. // Wait for completion.
  108. for (U32 i = 0; i < NUM_THREADS; i++)
  109. threads[i]->join();
  110. EXPECT_EQ(2 + ((1 + NUM_ADD_REFS_PER_THREAD + NUM_EXTRA_REFS_PER_THREAD) * NUM_THREADS * 2),
  111. mRef->getRefCount());
  112. // Run phase 2: release references.
  113. for (U32 i = 0; i < NUM_THREADS; i++)
  114. threads[i]->start((void*) 1);
  115. // Wait for completion.
  116. for (U32 i = 0; i < NUM_THREADS; i++)
  117. {
  118. threads[i]->join();
  119. delete threads[i];
  120. }
  121. EXPECT_EQ(2, mRef->getRefCount()); // increments of two
  122. mRef = NULL;
  123. }
  124. TEST_FIX(ThreadSafeRefCount, Tagging)
  125. {
  126. TestObjectRef ref;
  127. EXPECT_FALSE(ref.isTagged());
  128. EXPECT_FALSE(bool(ref));
  129. EXPECT_FALSE(bool(ref.ptr()));
  130. EXPECT_TRUE(ref.trySetFromTo(ref, NULL));
  131. EXPECT_FALSE(ref.isTagged());
  132. EXPECT_TRUE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_Set));
  133. EXPECT_TRUE(ref.isTagged());
  134. EXPECT_TRUE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_Set));
  135. EXPECT_TRUE(ref.isTagged());
  136. EXPECT_TRUE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_Unset));
  137. EXPECT_FALSE(ref.isTagged());
  138. EXPECT_TRUE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_Unset));
  139. EXPECT_FALSE(ref.isTagged());
  140. EXPECT_TRUE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_SetOrFail));
  141. EXPECT_TRUE(ref.isTagged());
  142. EXPECT_FALSE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_SetOrFail));
  143. EXPECT_TRUE(ref.isTagged());
  144. EXPECT_FALSE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_FailIfSet));
  145. EXPECT_TRUE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_UnsetOrFail));
  146. EXPECT_FALSE(ref.isTagged());
  147. EXPECT_FALSE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_UnsetOrFail));
  148. EXPECT_FALSE(ref.isTagged());
  149. EXPECT_FALSE(ref.trySetFromTo(ref, NULL, TestObjectRef::TAG_FailIfUnset));
  150. TestObjectRef objectA = new TestObject;
  151. TestObjectRef objectB = new TestObject;
  152. EXPECT_FALSE(objectA->isShared());
  153. EXPECT_FALSE(objectB->isShared());
  154. ref = objectA;
  155. EXPECT_FALSE(ref.isTagged());
  156. EXPECT_TRUE(ref == objectA);
  157. EXPECT_TRUE(ref == objectA.ptr());
  158. EXPECT_TRUE(objectA->isShared());
  159. EXPECT_TRUE(ref.trySetFromTo(objectA, objectB, TestObjectRef::TAG_Set));
  160. EXPECT_TRUE(ref.isTagged());
  161. EXPECT_EQ(ref, objectB);
  162. EXPECT_EQ(ref, objectB.ptr());
  163. EXPECT_TRUE(objectB->isShared());
  164. EXPECT_FALSE(objectA->isShared());
  165. EXPECT_TRUE(ref.trySetFromTo(ref, objectA));
  166. EXPECT_TRUE(ref.isTagged());
  167. EXPECT_EQ(ref, objectA);
  168. EXPECT_EQ(ref, objectA.ptr());
  169. }
  170. #endif