resource.cpp 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. #include <type_traits>
  2. #include <gtest/gtest.h>
  3. #include <entt/core/hashed_string.hpp>
  4. #include <entt/resource/cache.hpp>
  5. struct resource {
  6. int value;
  7. };
  8. struct derived_resource: resource {};
  9. template<typename Resource>
  10. struct loader: entt::resource_loader<loader<Resource>, Resource> {
  11. std::shared_ptr<Resource> load(int value) const {
  12. auto res = std::shared_ptr<Resource>(new Resource);
  13. res->value = value;
  14. return res;
  15. }
  16. };
  17. template<typename Resource>
  18. struct broken_loader: entt::resource_loader<broken_loader<Resource>, Resource> {
  19. std::shared_ptr<Resource> load(int) const {
  20. return nullptr;
  21. }
  22. };
  23. TEST(Resource, Functionalities) {
  24. entt::resource_cache<resource> cache;
  25. constexpr auto hs1 = entt::hashed_string{"res1"};
  26. constexpr auto hs2 = entt::hashed_string{"res2"};
  27. ASSERT_EQ(cache.size(), 0u);
  28. ASSERT_TRUE(cache.empty());
  29. ASSERT_FALSE(cache.contains(hs1));
  30. ASSERT_FALSE(cache.contains(hs2));
  31. ASSERT_FALSE(cache.load<broken_loader<resource>>(hs1, 42));
  32. ASSERT_FALSE(cache.reload<broken_loader<resource>>(hs1, 42));
  33. ASSERT_EQ(cache.size(), 0u);
  34. ASSERT_TRUE(cache.empty());
  35. ASSERT_FALSE(cache.contains(hs1));
  36. ASSERT_FALSE(cache.contains(hs2));
  37. ASSERT_TRUE(cache.load<loader<resource>>(hs1, 42));
  38. ASSERT_TRUE(cache.reload<loader<resource>>(hs1, 42));
  39. ASSERT_EQ(cache.handle(hs1).use_count(), 2);
  40. auto tmp = cache.handle(hs1);
  41. ASSERT_EQ(cache.handle(hs1).use_count(), 3);
  42. ASSERT_TRUE(static_cast<bool>(tmp));
  43. tmp = {};
  44. ASSERT_EQ(cache.handle(hs1).use_count(), 2);
  45. ASSERT_FALSE(static_cast<bool>(tmp));
  46. ASSERT_NE(cache.size(), 0u);
  47. ASSERT_FALSE(cache.empty());
  48. ASSERT_TRUE(cache.contains(hs1));
  49. ASSERT_FALSE(cache.contains(hs2));
  50. ASSERT_EQ((*cache.handle(hs1)).value, 42);
  51. ASSERT_TRUE(cache.load<loader<resource>>(hs1, 42));
  52. ASSERT_TRUE(cache.load<loader<resource>>(hs2, 42));
  53. ASSERT_NE(cache.size(), 0u);
  54. ASSERT_FALSE(cache.empty());
  55. ASSERT_TRUE(cache.contains(hs1));
  56. ASSERT_TRUE(cache.contains(hs2));
  57. ASSERT_EQ((*cache.handle(hs1)).value, 42);
  58. ASSERT_EQ(cache.handle(hs2)->value, 42);
  59. ASSERT_NO_FATAL_FAILURE(cache.discard(hs1));
  60. ASSERT_FALSE(cache.contains(hs1));
  61. ASSERT_TRUE(cache.contains(hs2));
  62. ASSERT_EQ(cache.handle(hs2)->value, 42);
  63. ASSERT_TRUE(cache.load<loader<resource>>(hs1, 42));
  64. ASSERT_NO_FATAL_FAILURE(cache.clear());
  65. ASSERT_EQ(cache.size(), 0u);
  66. ASSERT_TRUE(cache.empty());
  67. ASSERT_FALSE(cache.contains(hs1));
  68. ASSERT_FALSE(cache.contains(hs2));
  69. ASSERT_TRUE(cache.load<loader<resource>>(hs1, 42));
  70. ASSERT_NE(cache.size(), 0u);
  71. ASSERT_FALSE(cache.empty());
  72. ASSERT_TRUE(cache.handle(hs1));
  73. ASSERT_FALSE(cache.handle(hs2));
  74. ASSERT_TRUE(cache.handle(hs1));
  75. ASSERT_EQ(&cache.handle(hs1).get(), &static_cast<const resource &>(cache.handle(hs1)));
  76. ASSERT_NO_FATAL_FAILURE(cache.clear());
  77. ASSERT_EQ(cache.size(), 0u);
  78. ASSERT_TRUE(cache.empty());
  79. ASSERT_TRUE(cache.temp<loader<resource>>(42));
  80. ASSERT_TRUE(cache.empty());
  81. ASSERT_FALSE(entt::resource_handle<resource>{});
  82. ASSERT_TRUE(std::is_copy_constructible_v<entt::resource_handle<resource>>);
  83. ASSERT_TRUE(std::is_move_constructible_v<entt::resource_handle<resource>>);
  84. ASSERT_TRUE(std::is_copy_assignable_v<entt::resource_handle<resource>>);
  85. ASSERT_TRUE(std::is_move_assignable_v<entt::resource_handle<resource>>);
  86. }
  87. TEST(Resource, ConstNonConstHandle) {
  88. entt::resource_cache<resource> cache;
  89. entt::resource_handle<resource> handle = cache.temp<loader<resource>>(42);
  90. entt::resource_handle<const resource> chandle = handle;
  91. static_assert(std::is_same_v<decltype(handle.get()), resource &>);
  92. static_assert(std::is_same_v<decltype(chandle.get()), const resource &>);
  93. static_assert(std::is_same_v<decltype(std::as_const(handle).get()), resource &>);
  94. ASSERT_TRUE(chandle);
  95. ASSERT_EQ(handle.use_count(), 2u);
  96. ASSERT_EQ(chandle->value, 42);
  97. chandle = {};
  98. ASSERT_FALSE(chandle);
  99. ASSERT_EQ(handle.use_count(), 1u);
  100. }
  101. TEST(Resource, MutableHandle) {
  102. entt::resource_cache<resource> cache;
  103. constexpr auto hs = entt::hashed_string{"res"};
  104. auto handle = cache.load<loader<resource>>(hs, 0);
  105. ASSERT_TRUE(handle);
  106. ++handle.get().value;
  107. ++static_cast<resource &>(handle).value;
  108. ++(*handle).value;
  109. ++handle->value;
  110. ASSERT_EQ(cache.handle(hs)->value, 4);
  111. }
  112. TEST(Resource, HandleCast) {
  113. using namespace entt::literals;
  114. entt::resource_cache<resource> cache;
  115. auto handle = cache.load<loader<derived_resource>>("resource"_hs, 0);
  116. auto resource = std::make_shared<derived_resource>();
  117. entt::resource_handle<derived_resource> other{resource};
  118. ASSERT_TRUE(handle);
  119. ASSERT_TRUE(other);
  120. ASSERT_NE(&*handle, &*other);
  121. ASSERT_EQ(resource.use_count(), 2u);
  122. auto temp = std::move(handle);
  123. handle = other;
  124. ASSERT_TRUE(handle);
  125. ASSERT_TRUE(other);
  126. ASSERT_TRUE(temp);
  127. ASSERT_EQ(&*handle, &*other);
  128. ASSERT_EQ(resource.use_count(), 3u);
  129. temp = std::move(other);
  130. ASSERT_TRUE(handle);
  131. ASSERT_FALSE(other);
  132. ASSERT_TRUE(temp);
  133. ASSERT_EQ(&*handle, &*temp);
  134. ASSERT_EQ(resource.use_count(), 3u);
  135. temp = handle = {};
  136. ASSERT_FALSE(handle);
  137. ASSERT_FALSE(other);
  138. ASSERT_FALSE(temp);
  139. ASSERT_EQ(resource.use_count(), 1u);
  140. }
  141. TEST(Resource, Each) {
  142. using namespace entt::literals;
  143. entt::resource_cache<resource> cache;
  144. cache.load<loader<resource>>("resource"_hs, 0);
  145. cache.each([](entt::resource_handle<resource> res) {
  146. ++res->value;
  147. });
  148. ASSERT_FALSE(cache.empty());
  149. ASSERT_EQ(cache.handle("resource"_hs)->value, 1);
  150. cache.each([](auto id, auto res) {
  151. ASSERT_EQ(id, "resource"_hs);
  152. ++res->value;
  153. });
  154. ASSERT_FALSE(cache.empty());
  155. ASSERT_EQ(cache.handle("resource"_hs)->value, 2);
  156. cache.each([&cache](entt::id_type id) {
  157. cache.discard(id);
  158. });
  159. ASSERT_TRUE(cache.empty());
  160. }