2 * SPDX-License-Identifier: MIT
4 * Copyright © 2016 Intel Corporation
7 #include "i915_scatterlist.h"
8 #include "i915_ttm_buddy_manager.h"
10 #include <drm/drm_buddy.h>
11 #include <drm/drm_mm.h>
13 #include <linux/slab.h>
15 bool i915_sg_trim(struct sg_table *orig_st)
17 struct sg_table new_st;
18 struct scatterlist *sg, *new_sg;
21 if (orig_st->nents == orig_st->orig_nents)
24 if (sg_alloc_table(&new_st, orig_st->nents, GFP_KERNEL | __GFP_NOWARN))
28 for_each_sg(orig_st->sgl, sg, orig_st->nents, i) {
29 sg_set_page(new_sg, sg_page(sg), sg->length, 0);
30 sg_dma_address(new_sg) = sg_dma_address(sg);
31 sg_dma_len(new_sg) = sg_dma_len(sg);
33 new_sg = sg_next(new_sg);
35 GEM_BUG_ON(new_sg); /* Should walk exactly nents and hit the end */
37 sg_free_table(orig_st);
43 static void i915_refct_sgt_release(struct kref *ref)
45 struct i915_refct_sgt *rsgt =
46 container_of(ref, typeof(*rsgt), kref);
48 sg_free_table(&rsgt->table);
52 static const struct i915_refct_sgt_ops rsgt_ops = {
53 .release = i915_refct_sgt_release
57 * i915_refct_sgt_init - Initialize a struct i915_refct_sgt with default ops
58 * @rsgt: The struct i915_refct_sgt to initialize.
59 * size: The size of the underlying memory buffer.
61 void i915_refct_sgt_init(struct i915_refct_sgt *rsgt, size_t size)
63 __i915_refct_sgt_init(rsgt, size, &rsgt_ops);
67 * i915_rsgt_from_mm_node - Create a refcounted sg_table from a struct
69 * @node: The drm_mm_node.
70 * @region_start: An offset to add to the dma addresses of the sg list.
72 * Create a struct sg_table, initializing it from a struct drm_mm_node,
73 * taking a maximum segment length into account, splitting into segments
76 * Return: A pointer to a kmalloced struct i915_refct_sgt on success, negative
77 * error code cast to an error pointer on failure.
79 struct i915_refct_sgt *i915_rsgt_from_mm_node(const struct drm_mm_node *node,
82 const u64 max_segment = SZ_1G; /* Do we have a limit on this? */
83 u64 segment_pages = max_segment >> PAGE_SHIFT;
84 u64 block_size, offset, prev_end;
85 struct i915_refct_sgt *rsgt;
87 struct scatterlist *sg;
89 rsgt = kmalloc(sizeof(*rsgt), GFP_KERNEL);
91 return ERR_PTR(-ENOMEM);
93 i915_refct_sgt_init(rsgt, node->size << PAGE_SHIFT);
95 if (sg_alloc_table(st, DIV_ROUND_UP(node->size, segment_pages),
97 i915_refct_sgt_put(rsgt);
98 return ERR_PTR(-ENOMEM);
103 prev_end = (resource_size_t)-1;
104 block_size = node->size << PAGE_SHIFT;
105 offset = node->start << PAGE_SHIFT;
110 if (offset != prev_end || sg->length >= max_segment) {
114 sg_dma_address(sg) = region_start + offset;
120 len = min(block_size, max_segment - sg->length);
122 sg_dma_len(sg) += len;
137 * i915_rsgt_from_buddy_resource - Create a refcounted sg_table from a struct
138 * i915_buddy_block list
139 * @res: The struct i915_ttm_buddy_resource.
140 * @region_start: An offset to add to the dma addresses of the sg list.
142 * Create a struct sg_table, initializing it from struct i915_buddy_block list,
143 * taking a maximum segment length into account, splitting into segments
146 * Return: A pointer to a kmalloced struct i915_refct_sgts on success, negative
147 * error code cast to an error pointer on failure.
149 struct i915_refct_sgt *i915_rsgt_from_buddy_resource(struct ttm_resource *res,
152 struct i915_ttm_buddy_resource *bman_res = to_ttm_buddy_resource(res);
153 const u64 size = res->num_pages << PAGE_SHIFT;
154 const u64 max_segment = rounddown(UINT_MAX, PAGE_SIZE);
155 struct drm_buddy *mm = bman_res->mm;
156 struct list_head *blocks = &bman_res->blocks;
157 struct drm_buddy_block *block;
158 struct i915_refct_sgt *rsgt;
159 struct scatterlist *sg;
161 resource_size_t prev_end;
163 GEM_BUG_ON(list_empty(blocks));
165 rsgt = kmalloc(sizeof(*rsgt), GFP_KERNEL);
167 return ERR_PTR(-ENOMEM);
169 i915_refct_sgt_init(rsgt, size);
171 if (sg_alloc_table(st, res->num_pages, GFP_KERNEL)) {
172 i915_refct_sgt_put(rsgt);
173 return ERR_PTR(-ENOMEM);
178 prev_end = (resource_size_t)-1;
180 list_for_each_entry(block, blocks, link) {
181 u64 block_size, offset;
183 block_size = min_t(u64, size, drm_buddy_block_size(mm, block));
184 offset = drm_buddy_block_offset(block);
189 if (offset != prev_end || sg->length >= max_segment) {
193 sg_dma_address(sg) = region_start + offset;
199 len = min(block_size, max_segment - sg->length);
201 sg_dma_len(sg) += len;
216 #if IS_ENABLED(CONFIG_DRM_I915_SELFTEST)
217 #include "selftests/scatterlist.c"