ttm_mock_manager.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. // SPDX-License-Identifier: GPL-2.0 AND MIT
  2. /*
  3. * Copyright © 2023 Intel Corporation
  4. */
  5. #include <drm/ttm/ttm_resource.h>
  6. #include <drm/ttm/ttm_device.h>
  7. #include <drm/ttm/ttm_placement.h>
  8. #include "ttm_mock_manager.h"
  9. static inline struct ttm_mock_manager *
  10. to_mock_mgr(struct ttm_resource_manager *man)
  11. {
  12. return container_of(man, struct ttm_mock_manager, man);
  13. }
  14. static inline struct ttm_mock_resource *
  15. to_mock_mgr_resource(struct ttm_resource *res)
  16. {
  17. return container_of(res, struct ttm_mock_resource, base);
  18. }
  19. static int ttm_mock_manager_alloc(struct ttm_resource_manager *man,
  20. struct ttm_buffer_object *bo,
  21. const struct ttm_place *place,
  22. struct ttm_resource **res)
  23. {
  24. struct ttm_mock_manager *manager = to_mock_mgr(man);
  25. struct ttm_mock_resource *mock_res;
  26. struct drm_buddy *mm = &manager->mm;
  27. u64 lpfn, fpfn, alloc_size;
  28. int err;
  29. mock_res = kzalloc(sizeof(*mock_res), GFP_KERNEL);
  30. if (!mock_res)
  31. return -ENOMEM;
  32. fpfn = 0;
  33. lpfn = man->size;
  34. ttm_resource_init(bo, place, &mock_res->base);
  35. INIT_LIST_HEAD(&mock_res->blocks);
  36. if (place->flags & TTM_PL_FLAG_TOPDOWN)
  37. mock_res->flags |= DRM_BUDDY_TOPDOWN_ALLOCATION;
  38. if (place->flags & TTM_PL_FLAG_CONTIGUOUS)
  39. mock_res->flags |= DRM_BUDDY_CONTIGUOUS_ALLOCATION;
  40. alloc_size = (uint64_t)mock_res->base.size;
  41. mutex_lock(&manager->lock);
  42. err = drm_buddy_alloc_blocks(mm, fpfn, lpfn, alloc_size,
  43. manager->default_page_size,
  44. &mock_res->blocks,
  45. mock_res->flags);
  46. if (err)
  47. goto error_free_blocks;
  48. mutex_unlock(&manager->lock);
  49. *res = &mock_res->base;
  50. return 0;
  51. error_free_blocks:
  52. drm_buddy_free_list(mm, &mock_res->blocks, 0);
  53. ttm_resource_fini(man, &mock_res->base);
  54. mutex_unlock(&manager->lock);
  55. return err;
  56. }
  57. static void ttm_mock_manager_free(struct ttm_resource_manager *man,
  58. struct ttm_resource *res)
  59. {
  60. struct ttm_mock_manager *manager = to_mock_mgr(man);
  61. struct ttm_mock_resource *mock_res = to_mock_mgr_resource(res);
  62. struct drm_buddy *mm = &manager->mm;
  63. mutex_lock(&manager->lock);
  64. drm_buddy_free_list(mm, &mock_res->blocks, 0);
  65. mutex_unlock(&manager->lock);
  66. ttm_resource_fini(man, res);
  67. kfree(mock_res);
  68. }
  69. static const struct ttm_resource_manager_func ttm_mock_manager_funcs = {
  70. .alloc = ttm_mock_manager_alloc,
  71. .free = ttm_mock_manager_free,
  72. };
  73. int ttm_mock_manager_init(struct ttm_device *bdev, u32 mem_type, u32 size)
  74. {
  75. struct ttm_mock_manager *manager;
  76. struct ttm_resource_manager *base;
  77. int err;
  78. manager = kzalloc(sizeof(*manager), GFP_KERNEL);
  79. if (!manager)
  80. return -ENOMEM;
  81. mutex_init(&manager->lock);
  82. err = drm_buddy_init(&manager->mm, size, PAGE_SIZE);
  83. if (err) {
  84. kfree(manager);
  85. return err;
  86. }
  87. manager->default_page_size = PAGE_SIZE;
  88. base = &manager->man;
  89. base->func = &ttm_mock_manager_funcs;
  90. base->use_tt = true;
  91. ttm_resource_manager_init(base, bdev, size);
  92. ttm_set_driver_manager(bdev, mem_type, base);
  93. ttm_resource_manager_set_used(base, true);
  94. return 0;
  95. }
  96. EXPORT_SYMBOL_GPL(ttm_mock_manager_init);
  97. void ttm_mock_manager_fini(struct ttm_device *bdev, u32 mem_type)
  98. {
  99. struct ttm_resource_manager *man;
  100. struct ttm_mock_manager *mock_man;
  101. int err;
  102. man = ttm_manager_type(bdev, mem_type);
  103. mock_man = to_mock_mgr(man);
  104. err = ttm_resource_manager_evict_all(bdev, man);
  105. if (err)
  106. return;
  107. ttm_resource_manager_set_used(man, false);
  108. mutex_lock(&mock_man->lock);
  109. drm_buddy_fini(&mock_man->mm);
  110. mutex_unlock(&mock_man->lock);
  111. ttm_set_driver_manager(bdev, mem_type, NULL);
  112. }
  113. EXPORT_SYMBOL_GPL(ttm_mock_manager_fini);
  114. static int ttm_bad_manager_alloc(struct ttm_resource_manager *man,
  115. struct ttm_buffer_object *bo,
  116. const struct ttm_place *place,
  117. struct ttm_resource **res)
  118. {
  119. return -ENOSPC;
  120. }
  121. static int ttm_busy_manager_alloc(struct ttm_resource_manager *man,
  122. struct ttm_buffer_object *bo,
  123. const struct ttm_place *place,
  124. struct ttm_resource **res)
  125. {
  126. return -EBUSY;
  127. }
  128. static void ttm_bad_manager_free(struct ttm_resource_manager *man,
  129. struct ttm_resource *res)
  130. {
  131. }
  132. static bool ttm_bad_manager_compatible(struct ttm_resource_manager *man,
  133. struct ttm_resource *res,
  134. const struct ttm_place *place,
  135. size_t size)
  136. {
  137. return true;
  138. }
  139. static const struct ttm_resource_manager_func ttm_bad_manager_funcs = {
  140. .alloc = ttm_bad_manager_alloc,
  141. .free = ttm_bad_manager_free,
  142. .compatible = ttm_bad_manager_compatible
  143. };
  144. static const struct ttm_resource_manager_func ttm_bad_busy_manager_funcs = {
  145. .alloc = ttm_busy_manager_alloc,
  146. .free = ttm_bad_manager_free,
  147. .compatible = ttm_bad_manager_compatible
  148. };
  149. int ttm_bad_manager_init(struct ttm_device *bdev, u32 mem_type, u32 size)
  150. {
  151. struct ttm_resource_manager *man;
  152. man = kzalloc(sizeof(*man), GFP_KERNEL);
  153. if (!man)
  154. return -ENOMEM;
  155. man->func = &ttm_bad_manager_funcs;
  156. ttm_resource_manager_init(man, bdev, size);
  157. ttm_set_driver_manager(bdev, mem_type, man);
  158. ttm_resource_manager_set_used(man, true);
  159. return 0;
  160. }
  161. EXPORT_SYMBOL_GPL(ttm_bad_manager_init);
  162. int ttm_busy_manager_init(struct ttm_device *bdev, u32 mem_type, u32 size)
  163. {
  164. struct ttm_resource_manager *man;
  165. ttm_bad_manager_init(bdev, mem_type, size);
  166. man = ttm_manager_type(bdev, mem_type);
  167. man->func = &ttm_bad_busy_manager_funcs;
  168. return 0;
  169. }
  170. EXPORT_SYMBOL_GPL(ttm_busy_manager_init);
  171. void ttm_bad_manager_fini(struct ttm_device *bdev, uint32_t mem_type)
  172. {
  173. struct ttm_resource_manager *man;
  174. man = ttm_manager_type(bdev, mem_type);
  175. ttm_resource_manager_set_used(man, false);
  176. ttm_set_driver_manager(bdev, mem_type, NULL);
  177. kfree(man);
  178. }
  179. EXPORT_SYMBOL_GPL(ttm_bad_manager_fini);
  180. MODULE_DESCRIPTION("KUnit tests for ttm with mock resource managers");
  181. MODULE_LICENSE("GPL and additional rights");