1
0

resource.cpp 7.7 KB

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