Linux Audio

Check our new training course

Loading...
Note: File does not exist in v5.4.
  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
 18struct xe_ttm_sys_node {
 19	struct ttm_buffer_object *tbo;
 20	struct ttm_range_mgr_node base;
 21};
 22
 23static inline struct xe_ttm_sys_node *
 24to_xe_ttm_sys_node(struct ttm_resource *res)
 25{
 26	return container_of(res, struct xe_ttm_sys_node, base.base);
 27}
 28
 29static 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
 58err_fini:
 59	ttm_resource_fini(man, &node->base.base);
 60	kfree(node);
 61	return r;
 62}
 63
 64static 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
 73static void xe_ttm_sys_mgr_debug(struct ttm_resource_manager *man,
 74				 struct drm_printer *printer)
 75{
 76	/*
 77	 * This function is called by debugfs entry and would require
 78	 * pm_runtime_{get,put} wrappers around any operation.
 79	 */
 80}
 81
 82static const struct ttm_resource_manager_func xe_ttm_sys_mgr_func = {
 83	.alloc = xe_ttm_sys_mgr_new,
 84	.free = xe_ttm_sys_mgr_del,
 85	.debug = xe_ttm_sys_mgr_debug
 86};
 87
 88static void ttm_sys_mgr_fini(struct drm_device *drm, void *arg)
 89{
 90	struct xe_device *xe = (struct xe_device *)arg;
 91	struct ttm_resource_manager *man = &xe->mem.sys_mgr;
 92	int err;
 93
 94	ttm_resource_manager_set_used(man, false);
 95
 96	err = ttm_resource_manager_evict_all(&xe->ttm, man);
 97	if (err)
 98		return;
 99
100	ttm_resource_manager_cleanup(man);
101	ttm_set_driver_manager(&xe->ttm, XE_PL_TT, NULL);
102}
103
104int xe_ttm_sys_mgr_init(struct xe_device *xe)
105{
106	struct ttm_resource_manager *man = &xe->mem.sys_mgr;
107	struct sysinfo si;
108	u64 gtt_size;
109
110	si_meminfo(&si);
111	gtt_size = (u64)si.totalram * si.mem_unit;
112	/* TTM limits allocation of all TTM devices by 50% of system memory */
113	gtt_size /= 2;
114
115	man->use_tt = true;
116	man->func = &xe_ttm_sys_mgr_func;
117	ttm_resource_manager_init(man, &xe->ttm, gtt_size >> PAGE_SHIFT);
118	ttm_set_driver_manager(&xe->ttm, XE_PL_TT, man);
119	ttm_resource_manager_set_used(man, true);
120	return drmm_add_action_or_reset(&xe->drm, ttm_sys_mgr_fini, xe);
121}