Skip to content

Commit fe4d0b6

Browse files
freemangordontomba
authored andcommitted
drm: omapdrm: Export correct scatterlist for TILER backed BOs
Memory of BOs backed by TILER is not contiguous, but omap_gem_map_dma_buf() exports it like it is. This leads to (possibly) invalid memory accesses if another device imports such a BO. Fix that by providing sg that correctly describes TILER memory layout. Align TILER allocations to page, so importer to be able to correctly set its MMU if have one. Set export size accounting for the alignment. Also, make sure to destroy sg on unpin, as it is no longer valid. Tested on Motorola Droid4 by using GPU (sgx540) to render. Suggested-by: Matthijs van Duin <[email protected]> Signed-off-by: Ivaylo Dimitrov <[email protected]> Reviewed-by: Tomi Valkeinen <[email protected]> Signed-off-by: Tomi Valkeinen <[email protected]> Link: https://patchwork.freedesktop.org/patch/msgid/[email protected]
1 parent f8378c0 commit fe4d0b6

File tree

3 files changed

+85
-30
lines changed

3 files changed

+85
-30
lines changed

drivers/gpu/drm/omapdrm/omap_gem.c

Lines changed: 78 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -789,7 +789,7 @@ int omap_gem_pin(struct drm_gem_object *obj, dma_addr_t *dma_addr)
789789
if (omap_obj->flags & OMAP_BO_TILED_MASK) {
790790
block = tiler_reserve_2d(fmt,
791791
omap_obj->width,
792-
omap_obj->height, 0);
792+
omap_obj->height, PAGE_SIZE);
793793
} else {
794794
block = tiler_reserve_1d(obj->size);
795795
}
@@ -851,6 +851,11 @@ static void omap_gem_unpin_locked(struct drm_gem_object *obj)
851851
return;
852852

853853
if (refcount_dec_and_test(&omap_obj->dma_addr_cnt)) {
854+
if (omap_obj->sgt) {
855+
sg_free_table(omap_obj->sgt);
856+
kfree(omap_obj->sgt);
857+
omap_obj->sgt = NULL;
858+
}
854859
ret = tiler_unpin(omap_obj->block);
855860
if (ret) {
856861
dev_err(obj->dev->dev,
@@ -963,6 +968,78 @@ int omap_gem_put_pages(struct drm_gem_object *obj)
963968
return 0;
964969
}
965970

971+
struct sg_table *omap_gem_get_sg(struct drm_gem_object *obj)
972+
{
973+
struct omap_gem_object *omap_obj = to_omap_bo(obj);
974+
dma_addr_t addr;
975+
struct sg_table *sgt;
976+
struct scatterlist *sg;
977+
unsigned int count, len, stride, i;
978+
int ret;
979+
980+
ret = omap_gem_pin(obj, &addr);
981+
if (ret)
982+
return ERR_PTR(ret);
983+
984+
mutex_lock(&omap_obj->lock);
985+
986+
sgt = omap_obj->sgt;
987+
if (sgt)
988+
goto out;
989+
990+
sgt = kzalloc(sizeof(*sgt), GFP_KERNEL);
991+
if (!sgt) {
992+
ret = -ENOMEM;
993+
goto err_unpin;
994+
}
995+
996+
if (omap_obj->flags & OMAP_BO_TILED_MASK) {
997+
enum tiler_fmt fmt = gem2fmt(omap_obj->flags);
998+
999+
len = omap_obj->width << (int)fmt;
1000+
count = omap_obj->height;
1001+
stride = tiler_stride(fmt, 0);
1002+
} else {
1003+
len = obj->size;
1004+
count = 1;
1005+
stride = 0;
1006+
}
1007+
1008+
ret = sg_alloc_table(sgt, count, GFP_KERNEL);
1009+
if (ret)
1010+
goto err_free;
1011+
1012+
for_each_sg(sgt->sgl, sg, count, i) {
1013+
sg_set_page(sg, phys_to_page(addr), len, offset_in_page(addr));
1014+
sg_dma_address(sg) = addr;
1015+
sg_dma_len(sg) = len;
1016+
1017+
addr += stride;
1018+
}
1019+
1020+
omap_obj->sgt = sgt;
1021+
out:
1022+
mutex_unlock(&omap_obj->lock);
1023+
return sgt;
1024+
1025+
err_free:
1026+
kfree(sgt);
1027+
err_unpin:
1028+
mutex_unlock(&omap_obj->lock);
1029+
omap_gem_unpin(obj);
1030+
return ERR_PTR(ret);
1031+
}
1032+
1033+
void omap_gem_put_sg(struct drm_gem_object *obj, struct sg_table *sgt)
1034+
{
1035+
struct omap_gem_object *omap_obj = to_omap_bo(obj);
1036+
1037+
if (WARN_ON(omap_obj->sgt != sgt))
1038+
return;
1039+
1040+
omap_gem_unpin(obj);
1041+
}
1042+
9661043
#ifdef CONFIG_DRM_FBDEV_EMULATION
9671044
/*
9681045
* Get kernel virtual address for CPU access.. this more or less only

drivers/gpu/drm/omapdrm/omap_gem.h

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -82,5 +82,7 @@ u32 omap_gem_flags(struct drm_gem_object *obj);
8282
int omap_gem_rotated_dma_addr(struct drm_gem_object *obj, u32 orient,
8383
int x, int y, dma_addr_t *dma_addr);
8484
int omap_gem_tiled_stride(struct drm_gem_object *obj, u32 orient);
85+
struct sg_table *omap_gem_get_sg(struct drm_gem_object *obj);
86+
void omap_gem_put_sg(struct drm_gem_object *obj, struct sg_table *sgt);
8587

8688
#endif /* __OMAPDRM_GEM_H__ */

drivers/gpu/drm/omapdrm/omap_gem_dmabuf.c

Lines changed: 5 additions & 29 deletions
Original file line numberDiff line numberDiff line change
@@ -23,45 +23,21 @@ static struct sg_table *omap_gem_map_dma_buf(
2323
{
2424
struct drm_gem_object *obj = attachment->dmabuf->priv;
2525
struct sg_table *sg;
26-
dma_addr_t dma_addr;
27-
int ret;
28-
29-
sg = kzalloc(sizeof(*sg), GFP_KERNEL);
30-
if (!sg)
31-
return ERR_PTR(-ENOMEM);
32-
33-
/* camera, etc, need physically contiguous.. but we need a
34-
* better way to know this..
35-
*/
36-
ret = omap_gem_pin(obj, &dma_addr);
37-
if (ret)
38-
goto out;
39-
40-
ret = sg_alloc_table(sg, 1, GFP_KERNEL);
41-
if (ret)
42-
goto out;
43-
44-
sg_init_table(sg->sgl, 1);
45-
sg_dma_len(sg->sgl) = obj->size;
46-
sg_set_page(sg->sgl, pfn_to_page(PFN_DOWN(dma_addr)), obj->size, 0);
47-
sg_dma_address(sg->sgl) = dma_addr;
26+
sg = omap_gem_get_sg(obj);
27+
if (IS_ERR(sg))
28+
return sg;
4829

4930
/* this must be after omap_gem_pin() to ensure we have pages attached */
5031
omap_gem_dma_sync_buffer(obj, dir);
5132

5233
return sg;
53-
out:
54-
kfree(sg);
55-
return ERR_PTR(ret);
5634
}
5735

5836
static void omap_gem_unmap_dma_buf(struct dma_buf_attachment *attachment,
5937
struct sg_table *sg, enum dma_data_direction dir)
6038
{
6139
struct drm_gem_object *obj = attachment->dmabuf->priv;
62-
omap_gem_unpin(obj);
63-
sg_free_table(sg);
64-
kfree(sg);
40+
omap_gem_put_sg(obj, sg);
6541
}
6642

6743
static int omap_gem_dmabuf_begin_cpu_access(struct dma_buf *buffer,
@@ -114,7 +90,7 @@ struct dma_buf *omap_gem_prime_export(struct drm_gem_object *obj, int flags)
11490
DEFINE_DMA_BUF_EXPORT_INFO(exp_info);
11591

11692
exp_info.ops = &omap_dmabuf_ops;
117-
exp_info.size = obj->size;
93+
exp_info.size = omap_gem_mmap_size(obj);
11894
exp_info.flags = flags;
11995
exp_info.priv = obj;
12096

0 commit comments

Comments
 (0)