1 // SPDX-License-Identifier: MIT 2 /* 3 * Copyright © 2021-2022 Intel Corporation 4 * Copyright (C) 2021-2002 Red Hat 5 */ 6 7 #include "xe_ttm_sys_mgr.h" 8 9 #include <drm/drm_managed.h> 10 11 #include <drm/ttm/ttm_placement.h> 12 #include <drm/ttm/ttm_range_manager.h> 13 #include <drm/ttm/ttm_tt.h> 14 15 #include "xe_bo.h" 16 #include "xe_gt.h" 17 18 struct xe_ttm_sys_node { 19 struct ttm_buffer_object *tbo; 20 struct ttm_range_mgr_node base; 21 }; 22 23 static inline struct xe_ttm_sys_node * 24 to_xe_ttm_sys_node(struct ttm_resource *res) 25 { 26 return container_of(res, struct xe_ttm_sys_node, base.base); 27 } 28 29 static int xe_ttm_sys_mgr_new(struct ttm_resource_manager *man, 30 struct ttm_buffer_object *tbo, 31 const struct ttm_place *place, 32 struct ttm_resource **res) 33 { 34 struct xe_ttm_sys_node *node; 35 int r; 36 37 node = kzalloc(struct_size(node, base.mm_nodes, 1), GFP_KERNEL); 38 if (!node) 39 return -ENOMEM; 40 41 node->tbo = tbo; 42 ttm_resource_init(tbo, place, &node->base.base); 43 44 if (!(place->flags & TTM_PL_FLAG_TEMPORARY) && 45 ttm_resource_manager_usage(man) > (man->size << PAGE_SHIFT)) { 46 r = -ENOSPC; 47 goto err_fini; 48 } 49 50 node->base.mm_nodes[0].start = 0; 51 node->base.mm_nodes[0].size = PFN_UP(node->base.base.size); 52 node->base.base.start = XE_BO_INVALID_OFFSET; 53 54 *res = &node->base.base; 55 56 return 0; 57 58 err_fini: 59 ttm_resource_fini(man, &node->base.base); 60 kfree(node); 61 return r; 62 } 63 64 static void xe_ttm_sys_mgr_del(struct ttm_resource_manager *man, 65 struct ttm_resource *res) 66 { 67 struct xe_ttm_sys_node *node = to_xe_ttm_sys_node(res); 68 69 ttm_resource_fini(man, res); 70 kfree(node); 71 } 72 73 static void xe_ttm_sys_mgr_debug(struct ttm_resource_manager *man, 74 struct drm_printer *printer) 75 { 76 77 } 78 79 static const struct ttm_resource_manager_func xe_ttm_sys_mgr_func = { 80 .alloc = xe_ttm_sys_mgr_new, 81 .free = xe_ttm_sys_mgr_del, 82 .debug = xe_ttm_sys_mgr_debug 83 }; 84 85 static void ttm_sys_mgr_fini(struct drm_device *drm, void *arg) 86 { 87 struct xe_device *xe = (struct xe_device *)arg; 88 struct ttm_resource_manager *man = &xe->mem.sys_mgr; 89 int err; 90 91 ttm_resource_manager_set_used(man, false); 92 93 err = ttm_resource_manager_evict_all(&xe->ttm, man); 94 if (err) 95 return; 96 97 ttm_resource_manager_cleanup(man); 98 ttm_set_driver_manager(&xe->ttm, XE_PL_TT, NULL); 99 } 100 101 int xe_ttm_sys_mgr_init(struct xe_device *xe) 102 { 103 struct ttm_resource_manager *man = &xe->mem.sys_mgr; 104 struct sysinfo si; 105 u64 gtt_size; 106 107 si_meminfo(&si); 108 gtt_size = (u64)si.totalram * si.mem_unit; 109 /* TTM limits allocation of all TTM devices by 50% of system memory */ 110 gtt_size /= 2; 111 112 man->use_tt = true; 113 man->func = &xe_ttm_sys_mgr_func; 114 ttm_resource_manager_init(man, &xe->ttm, gtt_size >> PAGE_SHIFT); 115 ttm_set_driver_manager(&xe->ttm, XE_PL_TT, man); 116 ttm_resource_manager_set_used(man, true); 117 return drmm_add_action_or_reset(&xe->drm, ttm_sys_mgr_fini, xe); 118 } 119