Skip to content

Commit 9940d9d

Browse files
mszyprowdaeinki
authored andcommitted
drm/exynos: gem: Get rid of the internal 'pages' array
Internal pages array and scatter-list for them is not really needed for anything. FBDev emulation can simply rely on the DMA-mapping framework to create a proper kernel mapping for the buffer, while all other buffer use cases don't really need that array at all. Suggested-by: Christian König <[email protected]> Signed-off-by: Marek Szyprowski <[email protected]> Signed-off-by: Inki Dae <[email protected]>
1 parent 24f6fe7 commit 9940d9d

File tree

3 files changed

+42
-123
lines changed

3 files changed

+42
-123
lines changed

drivers/gpu/drm/exynos/exynos_drm_fbdev.c

Lines changed: 1 addition & 27 deletions
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,6 @@ static int exynos_drm_fbdev_update(struct drm_fb_helper *helper,
7676
struct fb_info *fbi;
7777
struct drm_framebuffer *fb = helper->fb;
7878
unsigned int size = fb->width * fb->height * fb->format->cpp[0];
79-
unsigned int nr_pages;
8079
unsigned long offset;
8180

8281
fbi = drm_fb_helper_alloc_fbi(helper);
@@ -90,16 +89,6 @@ static int exynos_drm_fbdev_update(struct drm_fb_helper *helper,
9089

9190
drm_fb_helper_fill_info(fbi, helper, sizes);
9291

93-
nr_pages = exynos_gem->size >> PAGE_SHIFT;
94-
95-
exynos_gem->kvaddr = (void __iomem *) vmap(exynos_gem->pages, nr_pages,
96-
VM_MAP, pgprot_writecombine(PAGE_KERNEL));
97-
if (!exynos_gem->kvaddr) {
98-
DRM_DEV_ERROR(to_dma_dev(helper->dev),
99-
"failed to map pages to kernel space.\n");
100-
return -EIO;
101-
}
102-
10392
offset = fbi->var.xoffset * fb->format->cpp[0];
10493
offset += fbi->var.yoffset * fb->pitches[0];
10594

@@ -133,18 +122,7 @@ static int exynos_drm_fbdev_create(struct drm_fb_helper *helper,
133122

134123
size = mode_cmd.pitches[0] * mode_cmd.height;
135124

136-
exynos_gem = exynos_drm_gem_create(dev, EXYNOS_BO_CONTIG, size);
137-
/*
138-
* If physically contiguous memory allocation fails and if IOMMU is
139-
* supported then try to get buffer from non physically contiguous
140-
* memory area.
141-
*/
142-
if (IS_ERR(exynos_gem) && is_drm_iommu_supported(dev)) {
143-
dev_warn(dev->dev, "contiguous FB allocation failed, falling back to non-contiguous\n");
144-
exynos_gem = exynos_drm_gem_create(dev, EXYNOS_BO_NONCONTIG,
145-
size);
146-
}
147-
125+
exynos_gem = exynos_drm_gem_create(dev, EXYNOS_BO_WC, size, true);
148126
if (IS_ERR(exynos_gem))
149127
return PTR_ERR(exynos_gem);
150128

@@ -229,12 +207,8 @@ int exynos_drm_fbdev_init(struct drm_device *dev)
229207
static void exynos_drm_fbdev_destroy(struct drm_device *dev,
230208
struct drm_fb_helper *fb_helper)
231209
{
232-
struct exynos_drm_fbdev *exynos_fbd = to_exynos_fbdev(fb_helper);
233-
struct exynos_drm_gem *exynos_gem = exynos_fbd->exynos_gem;
234210
struct drm_framebuffer *fb;
235211

236-
vunmap(exynos_gem->kvaddr);
237-
238212
/* release drm framebuffer and real buffer */
239213
if (fb_helper->fb && fb_helper->fb->funcs) {
240214
fb = fb_helper->fb;

drivers/gpu/drm/exynos/exynos_drm_gem.c

Lines changed: 37 additions & 87 deletions
Original file line numberDiff line numberDiff line change
@@ -17,90 +17,53 @@
1717
#include "exynos_drm_drv.h"
1818
#include "exynos_drm_gem.h"
1919

20-
static int exynos_drm_alloc_buf(struct exynos_drm_gem *exynos_gem)
20+
static int exynos_drm_alloc_buf(struct exynos_drm_gem *exynos_gem, bool kvmap)
2121
{
2222
struct drm_device *dev = exynos_gem->base.dev;
23-
unsigned long attr;
24-
unsigned int nr_pages;
25-
struct sg_table sgt;
26-
int ret = -ENOMEM;
23+
unsigned long attr = 0;
2724

2825
if (exynos_gem->dma_addr) {
2926
DRM_DEV_DEBUG_KMS(to_dma_dev(dev), "already allocated.\n");
3027
return 0;
3128
}
3229

33-
exynos_gem->dma_attrs = 0;
34-
3530
/*
3631
* if EXYNOS_BO_CONTIG, fully physically contiguous memory
3732
* region will be allocated else physically contiguous
3833
* as possible.
3934
*/
4035
if (!(exynos_gem->flags & EXYNOS_BO_NONCONTIG))
41-
exynos_gem->dma_attrs |= DMA_ATTR_FORCE_CONTIGUOUS;
36+
attr |= DMA_ATTR_FORCE_CONTIGUOUS;
4237

4338
/*
4439
* if EXYNOS_BO_WC or EXYNOS_BO_NONCACHABLE, writecombine mapping
4540
* else cachable mapping.
4641
*/
4742
if (exynos_gem->flags & EXYNOS_BO_WC ||
4843
!(exynos_gem->flags & EXYNOS_BO_CACHABLE))
49-
attr = DMA_ATTR_WRITE_COMBINE;
44+
attr |= DMA_ATTR_WRITE_COMBINE;
5045
else
51-
attr = DMA_ATTR_NON_CONSISTENT;
52-
53-
exynos_gem->dma_attrs |= attr;
54-
exynos_gem->dma_attrs |= DMA_ATTR_NO_KERNEL_MAPPING;
55-
56-
nr_pages = exynos_gem->size >> PAGE_SHIFT;
46+
attr |= DMA_ATTR_NON_CONSISTENT;
5747

58-
exynos_gem->pages = kvmalloc_array(nr_pages, sizeof(struct page *),
59-
GFP_KERNEL | __GFP_ZERO);
60-
if (!exynos_gem->pages) {
61-
DRM_DEV_ERROR(to_dma_dev(dev), "failed to allocate pages.\n");
62-
return -ENOMEM;
63-
}
48+
/* FBDev emulation requires kernel mapping */
49+
if (!kvmap)
50+
attr |= DMA_ATTR_NO_KERNEL_MAPPING;
6451

52+
exynos_gem->dma_attrs = attr;
6553
exynos_gem->cookie = dma_alloc_attrs(to_dma_dev(dev), exynos_gem->size,
6654
&exynos_gem->dma_addr, GFP_KERNEL,
6755
exynos_gem->dma_attrs);
6856
if (!exynos_gem->cookie) {
6957
DRM_DEV_ERROR(to_dma_dev(dev), "failed to allocate buffer.\n");
70-
goto err_free;
71-
}
72-
73-
ret = dma_get_sgtable_attrs(to_dma_dev(dev), &sgt, exynos_gem->cookie,
74-
exynos_gem->dma_addr, exynos_gem->size,
75-
exynos_gem->dma_attrs);
76-
if (ret < 0) {
77-
DRM_DEV_ERROR(to_dma_dev(dev), "failed to get sgtable.\n");
78-
goto err_dma_free;
79-
}
80-
81-
if (drm_prime_sg_to_page_addr_arrays(&sgt, exynos_gem->pages, NULL,
82-
nr_pages)) {
83-
DRM_DEV_ERROR(to_dma_dev(dev), "invalid sgtable.\n");
84-
ret = -EINVAL;
85-
goto err_sgt_free;
58+
return -ENOMEM;
8659
}
8760

88-
sg_free_table(&sgt);
61+
if (kvmap)
62+
exynos_gem->kvaddr = exynos_gem->cookie;
8963

9064
DRM_DEV_DEBUG_KMS(to_dma_dev(dev), "dma_addr(0x%lx), size(0x%lx)\n",
9165
(unsigned long)exynos_gem->dma_addr, exynos_gem->size);
92-
9366
return 0;
94-
95-
err_sgt_free:
96-
sg_free_table(&sgt);
97-
err_dma_free:
98-
dma_free_attrs(to_dma_dev(dev), exynos_gem->size, exynos_gem->cookie,
99-
exynos_gem->dma_addr, exynos_gem->dma_attrs);
100-
err_free:
101-
kvfree(exynos_gem->pages);
102-
103-
return ret;
10467
}
10568

10669
static void exynos_drm_free_buf(struct exynos_drm_gem *exynos_gem)
@@ -118,8 +81,6 @@ static void exynos_drm_free_buf(struct exynos_drm_gem *exynos_gem)
11881
dma_free_attrs(to_dma_dev(dev), exynos_gem->size, exynos_gem->cookie,
11982
(dma_addr_t)exynos_gem->dma_addr,
12083
exynos_gem->dma_attrs);
121-
122-
kvfree(exynos_gem->pages);
12384
}
12485

12586
static int exynos_drm_gem_handle_create(struct drm_gem_object *obj,
@@ -203,7 +164,8 @@ static struct exynos_drm_gem *exynos_drm_gem_init(struct drm_device *dev,
203164

204165
struct exynos_drm_gem *exynos_drm_gem_create(struct drm_device *dev,
205166
unsigned int flags,
206-
unsigned long size)
167+
unsigned long size,
168+
bool kvmap)
207169
{
208170
struct exynos_drm_gem *exynos_gem;
209171
int ret;
@@ -237,7 +199,7 @@ struct exynos_drm_gem *exynos_drm_gem_create(struct drm_device *dev,
237199
/* set memory type and cache attribute from user side. */
238200
exynos_gem->flags = flags;
239201

240-
ret = exynos_drm_alloc_buf(exynos_gem);
202+
ret = exynos_drm_alloc_buf(exynos_gem, kvmap);
241203
if (ret < 0) {
242204
drm_gem_object_release(&exynos_gem->base);
243205
kfree(exynos_gem);
@@ -254,7 +216,7 @@ int exynos_drm_gem_create_ioctl(struct drm_device *dev, void *data,
254216
struct exynos_drm_gem *exynos_gem;
255217
int ret;
256218

257-
exynos_gem = exynos_drm_gem_create(dev, args->flags, args->size);
219+
exynos_gem = exynos_drm_gem_create(dev, args->flags, args->size, false);
258220
if (IS_ERR(exynos_gem))
259221
return PTR_ERR(exynos_gem);
260222

@@ -365,7 +327,7 @@ int exynos_drm_gem_dumb_create(struct drm_file *file_priv,
365327
else
366328
flags = EXYNOS_BO_CONTIG | EXYNOS_BO_WC;
367329

368-
exynos_gem = exynos_drm_gem_create(dev, flags, args->size);
330+
exynos_gem = exynos_drm_gem_create(dev, flags, args->size, false);
369331
if (IS_ERR(exynos_gem)) {
370332
dev_warn(dev->dev, "FB allocation failed.\n");
371333
return PTR_ERR(exynos_gem);
@@ -442,11 +404,24 @@ struct drm_gem_object *exynos_drm_gem_prime_import(struct drm_device *dev,
442404
struct sg_table *exynos_drm_gem_prime_get_sg_table(struct drm_gem_object *obj)
443405
{
444406
struct exynos_drm_gem *exynos_gem = to_exynos_gem(obj);
445-
int npages;
407+
struct drm_device *drm_dev = obj->dev;
408+
struct sg_table *sgt;
409+
int ret;
410+
411+
sgt = kzalloc(sizeof(*sgt), GFP_KERNEL);
412+
if (!sgt)
413+
return ERR_PTR(-ENOMEM);
446414

447-
npages = exynos_gem->size >> PAGE_SHIFT;
415+
ret = dma_get_sgtable_attrs(to_dma_dev(drm_dev), sgt, exynos_gem->cookie,
416+
exynos_gem->dma_addr, exynos_gem->size,
417+
exynos_gem->dma_attrs);
418+
if (ret) {
419+
DRM_ERROR("failed to get sgtable, %d\n", ret);
420+
kfree(sgt);
421+
return ERR_PTR(ret);
422+
}
448423

449-
return drm_prime_pages_to_sg(exynos_gem->pages, npages);
424+
return sgt;
450425
}
451426

452427
struct drm_gem_object *
@@ -455,8 +430,6 @@ exynos_drm_gem_prime_import_sg_table(struct drm_device *dev,
455430
struct sg_table *sgt)
456431
{
457432
struct exynos_drm_gem *exynos_gem;
458-
int npages;
459-
int ret;
460433

461434
if (sgt->nents < 1)
462435
return ERR_PTR(-EINVAL);
@@ -482,26 +455,8 @@ exynos_drm_gem_prime_import_sg_table(struct drm_device *dev,
482455
}
483456

484457
exynos_gem = exynos_drm_gem_init(dev, attach->dmabuf->size);
485-
if (IS_ERR(exynos_gem)) {
486-
ret = PTR_ERR(exynos_gem);
487-
return ERR_PTR(ret);
488-
}
489-
490-
exynos_gem->dma_addr = sg_dma_address(sgt->sgl);
491-
492-
npages = exynos_gem->size >> PAGE_SHIFT;
493-
exynos_gem->pages = kvmalloc_array(npages, sizeof(struct page *), GFP_KERNEL);
494-
if (!exynos_gem->pages) {
495-
ret = -ENOMEM;
496-
goto err;
497-
}
498-
499-
ret = drm_prime_sg_to_page_addr_arrays(sgt, exynos_gem->pages, NULL,
500-
npages);
501-
if (ret < 0)
502-
goto err_free_large;
503-
504-
exynos_gem->sgt = sgt;
458+
if (IS_ERR(exynos_gem))
459+
return ERR_CAST(exynos_gem);
505460

506461
/*
507462
* Buffer has been mapped as contiguous into DMA address space,
@@ -513,14 +468,9 @@ exynos_drm_gem_prime_import_sg_table(struct drm_device *dev,
513468
else
514469
exynos_gem->flags |= EXYNOS_BO_CONTIG;
515470

471+
exynos_gem->dma_addr = sg_dma_address(sgt->sgl);
472+
exynos_gem->sgt = sgt;
516473
return &exynos_gem->base;
517-
518-
err_free_large:
519-
kvfree(exynos_gem->pages);
520-
err:
521-
drm_gem_object_release(&exynos_gem->base);
522-
kfree(exynos_gem);
523-
return ERR_PTR(ret);
524474
}
525475

526476
void *exynos_drm_gem_prime_vmap(struct drm_gem_object *obj)

drivers/gpu/drm/exynos/exynos_drm_gem.h

Lines changed: 4 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -21,20 +21,15 @@
2121
* @base: a gem object.
2222
* - a new handle to this gem object would be created
2323
* by drm_gem_handle_create().
24-
* @buffer: a pointer to exynos_drm_gem_buffer object.
25-
* - contain the information to memory region allocated
26-
* by user request or at framebuffer creation.
27-
* continuous memory region allocated by user request
28-
* or at framebuffer creation.
2924
* @flags: indicate memory type to allocated buffer and cache attruibute.
3025
* @size: size requested from user, in bytes and this size is aligned
3126
* in page unit.
3227
* @cookie: cookie returned by dma_alloc_attrs
33-
* @kvaddr: kernel virtual address to allocated memory region.
28+
* @kvaddr: kernel virtual address to allocated memory region (for fbdev)
3429
* @dma_addr: bus address(accessed by dma) to allocated memory region.
3530
* - this address could be physical address without IOMMU and
3631
* device address with IOMMU.
37-
* @pages: Array of backing pages.
32+
* @dma_attrs: attrs passed dma mapping framework
3833
* @sgt: Imported sg_table.
3934
*
4035
* P.S. this object would be transferred to user as kms_bo.handle so
@@ -48,7 +43,6 @@ struct exynos_drm_gem {
4843
void __iomem *kvaddr;
4944
dma_addr_t dma_addr;
5045
unsigned long dma_attrs;
51-
struct page **pages;
5246
struct sg_table *sgt;
5347
};
5448

@@ -58,7 +52,8 @@ void exynos_drm_gem_destroy(struct exynos_drm_gem *exynos_gem);
5852
/* create a new buffer with gem object */
5953
struct exynos_drm_gem *exynos_drm_gem_create(struct drm_device *dev,
6054
unsigned int flags,
61-
unsigned long size);
55+
unsigned long size,
56+
bool kvmap);
6257

6358
/*
6459
* request gem object creation and buffer allocation as the size

0 commit comments

Comments
 (0)