340 lines
10 KiB
Diff
340 lines
10 KiB
Diff
From e6e383f45f59fdd935a2b13b1e9b00968ac5a1aa Mon Sep 17 00:00:00 2001
|
|
From: Myy <myy@miouyouyou.fr>
|
|
Date: Sun, 30 Jul 2017 17:32:26 +0000
|
|
Subject: [PATCH 2/9] Rockchip DRM: GEM Prime import SG Table Support
|
|
|
|
A combination of an old patch provided by the ARM developers and the
|
|
patches provided by Rockchip and @wzyy2.
|
|
|
|
This patch provide a Rockchip specific implementation for the Rockchip
|
|
DRM driver gem_prime_import_sg_table call.
|
|
|
|
Signed-off-by: Myy <myy@miouyouyou.fr>
|
|
---
|
|
drivers/gpu/drm/rockchip/rockchip_drm_drv.c | 10 +++
|
|
drivers/gpu/drm/rockchip/rockchip_drm_gem.c | 133 ++++++++++++++++++++++++++--
|
|
drivers/gpu/drm/rockchip/rockchip_drm_gem.h | 9 +-
|
|
include/uapi/drm/rockchip_drm.h | 57 ++++++++++++
|
|
4 files changed, 199 insertions(+), 10 deletions(-)
|
|
create mode 100644 include/uapi/drm/rockchip_drm.h
|
|
|
|
diff --git a/drivers/gpu/drm/rockchip/rockchip_drm_drv.c b/drivers/gpu/drm/rockchip/rockchip_drm_drv.c
|
|
index c6b1b7f3..a261948f 100644
|
|
--- a/drivers/gpu/drm/rockchip/rockchip_drm_drv.c
|
|
+++ b/drivers/gpu/drm/rockchip/rockchip_drm_drv.c
|
|
@@ -19,6 +19,7 @@
|
|
#include <drm/drm_fb_helper.h>
|
|
#include <drm/drm_gem_cma_helper.h>
|
|
#include <drm/drm_of.h>
|
|
+#include <drm/rockchip_drm.h>
|
|
#include <linux/dma-mapping.h>
|
|
#include <linux/dma-iommu.h>
|
|
#include <linux/pm_runtime.h>
|
|
@@ -217,6 +218,12 @@ static void rockchip_drm_lastclose(struct drm_device *dev)
|
|
drm_fb_helper_restore_fbdev_mode_unlocked(&priv->fbdev_helper);
|
|
}
|
|
|
|
+static const struct drm_ioctl_desc rockchip_ioctls[] = {
|
|
+ DRM_IOCTL_DEF_DRV(ROCKCHIP_GEM_CREATE, rockchip_gem_create_ioctl,
|
|
+ DRM_UNLOCKED | DRM_AUTH),
|
|
+};
|
|
+
|
|
+
|
|
static const struct file_operations rockchip_drm_driver_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = drm_open,
|
|
@@ -239,12 +246,15 @@ static struct drm_driver rockchip_drm_driver = {
|
|
.dumb_destroy = drm_gem_dumb_destroy,
|
|
.prime_handle_to_fd = drm_gem_prime_handle_to_fd,
|
|
.prime_fd_to_handle = drm_gem_prime_fd_to_handle,
|
|
+ .gem_prime_import_sg_table = rockchip_gem_prime_import_sg_table,
|
|
.gem_prime_import = drm_gem_prime_import,
|
|
.gem_prime_export = drm_gem_prime_export,
|
|
.gem_prime_get_sg_table = rockchip_gem_prime_get_sg_table,
|
|
.gem_prime_vmap = rockchip_gem_prime_vmap,
|
|
.gem_prime_vunmap = rockchip_gem_prime_vunmap,
|
|
.gem_prime_mmap = rockchip_gem_mmap_buf,
|
|
+ .ioctls = rockchip_ioctls,
|
|
+ .num_ioctls = ARRAY_SIZE(rockchip_ioctls),
|
|
.fops = &rockchip_drm_driver_fops,
|
|
.name = DRIVER_NAME,
|
|
.desc = DRIVER_DESC,
|
|
diff --git a/drivers/gpu/drm/rockchip/rockchip_drm_gem.c b/drivers/gpu/drm/rockchip/rockchip_drm_gem.c
|
|
index b74ac717..05dfdbed 100644
|
|
--- a/drivers/gpu/drm/rockchip/rockchip_drm_gem.c
|
|
+++ b/drivers/gpu/drm/rockchip/rockchip_drm_gem.c
|
|
@@ -16,7 +16,12 @@
|
|
#include <drm/drmP.h>
|
|
#include <drm/drm_gem.h>
|
|
#include <drm/drm_vma_manager.h>
|
|
+#include <drm/rockchip_drm.h>
|
|
+
|
|
#include <linux/iommu.h>
|
|
+#include <linux/dma-mapping.h>
|
|
+#include <linux/dma-buf.h>
|
|
+
|
|
|
|
#include "rockchip_drm_drv.h"
|
|
#include "rockchip_drm_gem.h"
|
|
@@ -309,12 +314,10 @@ static void rockchip_gem_release_object(struct rockchip_gem_object *rk_obj)
|
|
}
|
|
|
|
struct rockchip_gem_object *
|
|
- rockchip_gem_create_object(struct drm_device *drm, unsigned int size,
|
|
- bool alloc_kmap)
|
|
+rockchip_gem_alloc_object(struct drm_device *drm, unsigned int size)
|
|
{
|
|
struct rockchip_gem_object *rk_obj;
|
|
struct drm_gem_object *obj;
|
|
- int ret;
|
|
|
|
size = round_up(size, PAGE_SIZE);
|
|
|
|
@@ -325,6 +328,19 @@ struct rockchip_gem_object *
|
|
obj = &rk_obj->base;
|
|
|
|
drm_gem_object_init(drm, obj, size);
|
|
+ return rk_obj;
|
|
+}
|
|
+
|
|
+struct rockchip_gem_object *
|
|
+rockchip_gem_create_object(struct drm_device *drm, unsigned int size,
|
|
+ bool alloc_kmap)
|
|
+{
|
|
+ struct rockchip_gem_object *rk_obj;
|
|
+ int ret;
|
|
+
|
|
+ rk_obj = rockchip_gem_alloc_object(drm, size);
|
|
+ if (IS_ERR(rk_obj))
|
|
+ return rk_obj;
|
|
|
|
ret = rockchip_gem_alloc_buf(rk_obj, alloc_kmap);
|
|
if (ret)
|
|
@@ -343,13 +359,24 @@ struct rockchip_gem_object *
|
|
*/
|
|
void rockchip_gem_free_object(struct drm_gem_object *obj)
|
|
{
|
|
- struct rockchip_gem_object *rk_obj;
|
|
-
|
|
- rk_obj = to_rockchip_obj(obj);
|
|
+ struct drm_device *drm = obj->dev;
|
|
+ struct rockchip_drm_private *private = drm->dev_private;
|
|
+ struct rockchip_gem_object *rk_obj = to_rockchip_obj(obj);
|
|
|
|
- rockchip_gem_free_buf(rk_obj);
|
|
+ if (obj->import_attach) {
|
|
+ if (private->domain) {
|
|
+ rockchip_gem_iommu_unmap(rk_obj);
|
|
+ } else {
|
|
+ dma_unmap_sg(
|
|
+ drm->dev, rk_obj->sgt->sgl,
|
|
+ rk_obj->sgt->nents, DMA_BIDIRECTIONAL
|
|
+ );
|
|
+ }
|
|
+ drm_prime_gem_destroy(obj, rk_obj->sgt);
|
|
+ } else {
|
|
+ rockchip_gem_free_buf(rk_obj);
|
|
+ }
|
|
|
|
- rockchip_gem_release_object(rk_obj);
|
|
}
|
|
|
|
/*
|
|
@@ -445,6 +472,16 @@ int rockchip_gem_dumb_create(struct drm_file *file_priv,
|
|
return PTR_ERR_OR_ZERO(rk_obj);
|
|
}
|
|
|
|
+int rockchip_gem_create_ioctl(struct drm_device *dev, void *data,
|
|
+ struct drm_file *file_priv) {
|
|
+ struct drm_rockchip_gem_create *args = data;
|
|
+ struct rockchip_gem_object *rk_obj;
|
|
+
|
|
+ rk_obj = rockchip_gem_create_with_handle(file_priv, dev, args->size,
|
|
+ &args->handle);
|
|
+ return PTR_ERR_OR_ZERO(rk_obj);
|
|
+}
|
|
+
|
|
/*
|
|
* Allocate a sg_table for this GEM object.
|
|
* Note: Both the table's contents, and the sg_table itself must be freed by
|
|
@@ -477,6 +514,86 @@ struct sg_table *rockchip_gem_prime_get_sg_table(struct drm_gem_object *obj)
|
|
return sgt;
|
|
}
|
|
|
|
+static unsigned long rockchip_sg_get_contiguous_size(struct sg_table *sgt,
|
|
+ int count)
|
|
+{
|
|
+ struct scatterlist *s;
|
|
+ dma_addr_t expected = sg_dma_address(sgt->sgl);
|
|
+ unsigned int i;
|
|
+ unsigned long size = 0;
|
|
+
|
|
+ for_each_sg(sgt->sgl, s, count, i) {
|
|
+ if (sg_dma_address(s) != expected)
|
|
+ break;
|
|
+ expected = sg_dma_address(s) + sg_dma_len(s);
|
|
+ size += sg_dma_len(s);
|
|
+ }
|
|
+ return size;
|
|
+}
|
|
+
|
|
+static int
|
|
+rockchip_gem_iommu_map_sg(struct drm_device *drm,
|
|
+ struct dma_buf_attachment *attach,
|
|
+ struct sg_table *sg,
|
|
+ struct rockchip_gem_object *rk_obj)
|
|
+{
|
|
+ rk_obj->sgt = sg;
|
|
+ return rockchip_gem_iommu_map(rk_obj);
|
|
+}
|
|
+
|
|
+static int
|
|
+rockchip_gem_dma_map_sg(struct drm_device *drm,
|
|
+ struct dma_buf_attachment *attach,
|
|
+ struct sg_table *sg,
|
|
+ struct rockchip_gem_object *rk_obj)
|
|
+{
|
|
+ int count = dma_map_sg(drm->dev, sg->sgl, sg->nents,
|
|
+ DMA_BIDIRECTIONAL);
|
|
+ if (!count)
|
|
+ return -EINVAL;
|
|
+
|
|
+ if (rockchip_sg_get_contiguous_size(sg, count) < attach->dmabuf->size) {
|
|
+ DRM_ERROR("failed to map sg_table to contiguous linear address.\n");
|
|
+ dma_unmap_sg(drm->dev, sg->sgl, sg->nents,
|
|
+ DMA_BIDIRECTIONAL);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ rk_obj->dma_addr = sg_dma_address(sg->sgl);
|
|
+ rk_obj->sgt = sg;
|
|
+ return 0;
|
|
+}
|
|
+
|
|
+struct drm_gem_object *
|
|
+rockchip_gem_prime_import_sg_table(struct drm_device *drm,
|
|
+ struct dma_buf_attachment *attach,
|
|
+ struct sg_table *sg)
|
|
+{
|
|
+ struct rockchip_drm_private *private = drm->dev_private;
|
|
+ struct rockchip_gem_object *rk_obj;
|
|
+ int ret;
|
|
+
|
|
+ rk_obj = rockchip_gem_alloc_object(drm, attach->dmabuf->size);
|
|
+ if (IS_ERR(rk_obj))
|
|
+ return ERR_CAST(rk_obj);
|
|
+
|
|
+ if (private->domain)
|
|
+ ret = rockchip_gem_iommu_map_sg(drm, attach, sg, rk_obj);
|
|
+ else
|
|
+ ret = rockchip_gem_dma_map_sg(drm, attach, sg, rk_obj);
|
|
+
|
|
+ if (ret < 0) {
|
|
+ DRM_ERROR("failed to import sg table: %d\n", ret);
|
|
+ goto err_free_rk_obj;
|
|
+ }
|
|
+
|
|
+ return &rk_obj->base;
|
|
+
|
|
+err_free_rk_obj:
|
|
+ rockchip_gem_release_object(rk_obj);
|
|
+ return ERR_PTR(ret);
|
|
+}
|
|
+
|
|
void *rockchip_gem_prime_vmap(struct drm_gem_object *obj)
|
|
{
|
|
struct rockchip_gem_object *rk_obj = to_rockchip_obj(obj);
|
|
diff --git a/drivers/gpu/drm/rockchip/rockchip_drm_gem.h b/drivers/gpu/drm/rockchip/rockchip_drm_gem.h
|
|
index 3f6ea4d1..b8fed4df 100644
|
|
--- a/drivers/gpu/drm/rockchip/rockchip_drm_gem.h
|
|
+++ b/drivers/gpu/drm/rockchip/rockchip_drm_gem.h
|
|
@@ -36,8 +36,9 @@ struct rockchip_gem_object {
|
|
|
|
struct sg_table *rockchip_gem_prime_get_sg_table(struct drm_gem_object *obj);
|
|
struct drm_gem_object *
|
|
-rockchip_gem_prime_import_sg_table(struct drm_device *dev, size_t size,
|
|
- struct sg_table *sgt);
|
|
+rockchip_gem_prime_import_sg_table(struct drm_device *dev,
|
|
+ struct dma_buf_attachment *attach,
|
|
+ struct sg_table *sgt);
|
|
void *rockchip_gem_prime_vmap(struct drm_gem_object *obj);
|
|
void rockchip_gem_prime_vunmap(struct drm_gem_object *obj, void *vaddr);
|
|
|
|
@@ -57,6 +58,10 @@ void rockchip_gem_free_object(struct drm_gem_object *obj);
|
|
int rockchip_gem_dumb_create(struct drm_file *file_priv,
|
|
struct drm_device *dev,
|
|
struct drm_mode_create_dumb *args);
|
|
+int rockchip_gem_create_ioctl(struct drm_device *dev, void *data,
|
|
+ struct drm_file *file_priv);
|
|
+int rockchip_gem_map_offset_ioctl(struct drm_device *dev, void *data,
|
|
+ struct drm_file *file_priv);
|
|
int rockchip_gem_dumb_map_offset(struct drm_file *file_priv,
|
|
struct drm_device *dev, uint32_t handle,
|
|
uint64_t *offset);
|
|
diff --git a/include/uapi/drm/rockchip_drm.h b/include/uapi/drm/rockchip_drm.h
|
|
new file mode 100644
|
|
index 00000000..c521e5a3
|
|
--- /dev/null
|
|
+++ b/include/uapi/drm/rockchip_drm.h
|
|
@@ -0,0 +1,57 @@
|
|
+/*
|
|
+ *
|
|
+ * Copyright (c) Fuzhou Rockchip Electronics Co.Ltd
|
|
+ * Authors:
|
|
+ * Mark Yao <yzq@rock-chips.com>
|
|
+ *
|
|
+ * base on exynos_drm.h
|
|
+ *
|
|
+ * This program is free software; you can redistribute it and/or modify it
|
|
+ * under the terms of the GNU General Public License as published by the
|
|
+ * Free Software Foundation; either version 2 of the License, or (at your
|
|
+ * option) any later version.
|
|
+ */
|
|
+
|
|
+#ifndef _UAPI_ROCKCHIP_DRM_H
|
|
+#define _UAPI_ROCKCHIP_DRM_H
|
|
+
|
|
+#include <drm/drm.h>
|
|
+
|
|
+/**
|
|
+ * User-desired buffer creation information structure.
|
|
+ *
|
|
+ * @size: user-desired memory allocation size.
|
|
+ * @flags: user request for setting memory type or cache attributes.
|
|
+ * @handle: returned a handle to created gem object.
|
|
+ * - this handle will be set by gem module of kernel side.
|
|
+ */
|
|
+struct drm_rockchip_gem_create {
|
|
+ uint64_t size;
|
|
+ uint32_t flags;
|
|
+ uint32_t handle;
|
|
+};
|
|
+
|
|
+/**
|
|
+ * A structure for getting buffer offset.
|
|
+ *
|
|
+ * @handle: a pointer to gem object created.
|
|
+ * @pad: just padding to be 64-bit aligned.
|
|
+ * @offset: relatived offset value of the memory region allocated.
|
|
+ * - this value should be set by user.
|
|
+ */
|
|
+struct drm_rockchip_gem_map_off {
|
|
+ uint32_t handle;
|
|
+ uint32_t pad;
|
|
+ uint64_t offset;
|
|
+};
|
|
+
|
|
+#define DRM_ROCKCHIP_GEM_CREATE 0x00
|
|
+#define DRM_ROCKCHIP_GEM_MAP_OFFSET 0x01
|
|
+
|
|
+#define DRM_IOCTL_ROCKCHIP_GEM_CREATE DRM_IOWR(DRM_COMMAND_BASE + \
|
|
+ DRM_ROCKCHIP_GEM_CREATE, struct drm_rockchip_gem_create)
|
|
+
|
|
+#define DRM_IOCTL_ROCKCHIP_GEM_MAP_OFFSET DRM_IOWR(DRM_COMMAND_BASE + \
|
|
+ DRM_ROCKCHIP_GEM_MAP_OFFSET, struct drm_rockchip_gem_map_off)
|
|
+
|
|
+#endif /* _UAPI_ROCKCHIP_DRM_H */
|
|
--
|
|
2.13.0
|
|
|