[PATCH 2/6] drm/exynos: ipp: Add IPP v2 framework

Tobias Jakobi tjakobi at math.uni-bielefeld.de
Fri Sep 15 17:18:15 UTC 2017


Hello Marek,


Marek Szyprowski wrote:
> This patch adds Exynos IPP v2 subsystem and userspace API.
> 
> New userspace API is focused ONLY on memory-to-memory image processing.
> The two remainging IPP operation modes (framebuffer writeback and
> local-path output with image processing) can be implemented using
> standard DRM features: writeback connectors and additional DRM planes with
> scaling features.
> 
> V2 IPP userspace API is not compatible with old IPP ioctls. This is a
> significant change, but the old IPP subsystem in mainline Linux kernel was
> partially disfunctional anyway and not used in any open-source project.
> 
> V2 IPP userspace API is based on stateless approach, which much better fits
> to memory-to-memory image processing model. It also provides support for
> all image formats, which are both already defined in DRM API and supported
> by the existing IPP hardware modules.
> 
> The API consists of the following ioctls:
> - DRM_IOCTL_EXYNOS_IPP_GET_RESOURCES: to enumerate all available image
>   processing modules,
> - DRM_IOCTL_EXYNOS_IPP_GET_CAPS: to query capabilities and supported image
>   formats of given IPP module,
> - DRM_IOCTL_EXYNOS_IPP_GET_LIMITS: to query hardware limitiations for
>   selected image format of given IPP module,
> - DRM_IOCTL_EXYNOS_IPP_COMMIT: to perform operation described by the
>   provided structures (source and destination buffers, operation rectangle,
>   transformation, etc).
> 
> The proposed userspace API is extensible. In the future more advanced image
> processing operations can be defined to support for example blending.
> 
> Userspace API is fully functional also on DRM render nodes, so it is not
> limited to the root/privileged client.
> 
> Internal driver API also has been completely rewritten. New IPP core
> performs all possible input validation, checks and object life-time
> control. The drivers can focus only on writing configuration to hardware
> registers. Stateless nature of DRM_IOCTL_EXYNOS_IPP_COMMIT ioctl simplifies
> the driver API. Minimal driver needs to provide a single callback for
> starting processing and an array with supported image formats.
> 
> Signed-off-by: Marek Szyprowski <m.szyprowski at samsung.com>
> ---
>  drivers/gpu/drm/exynos/Kconfig          |   3 +
>  drivers/gpu/drm/exynos/Makefile         |   1 +
>  drivers/gpu/drm/exynos/exynos_drm_drv.c |   9 +
>  drivers/gpu/drm/exynos/exynos_drm_ipp.c | 893 ++++++++++++++++++++++++++++++++
>  drivers/gpu/drm/exynos/exynos_drm_ipp.h | 188 +++++++
>  include/uapi/drm/exynos_drm.h           | 227 ++++++++
>  6 files changed, 1321 insertions(+)
>  create mode 100644 drivers/gpu/drm/exynos/exynos_drm_ipp.c
>  create mode 100644 drivers/gpu/drm/exynos/exynos_drm_ipp.h
> 
> diff --git a/drivers/gpu/drm/exynos/Kconfig b/drivers/gpu/drm/exynos/Kconfig
> index 88cff0e039b6..2e097a81df12 100644
> --- a/drivers/gpu/drm/exynos/Kconfig
> +++ b/drivers/gpu/drm/exynos/Kconfig
> @@ -94,6 +94,9 @@ config DRM_EXYNOS_G2D
>  	help
>  	  Choose this option if you want to use Exynos G2D for DRM.
>  
> +config DRM_EXYNOS_IPP
> +	bool
> +
>  config DRM_EXYNOS_FIMC
>  	bool "FIMC"
>  	depends on BROKEN && MFD_SYSCON
> diff --git a/drivers/gpu/drm/exynos/Makefile b/drivers/gpu/drm/exynos/Makefile
> index 09bb002c9555..f663490e949d 100644
> --- a/drivers/gpu/drm/exynos/Makefile
> +++ b/drivers/gpu/drm/exynos/Makefile
> @@ -17,6 +17,7 @@ exynosdrm-$(CONFIG_DRM_EXYNOS_MIXER)	+= exynos_mixer.o
>  exynosdrm-$(CONFIG_DRM_EXYNOS_HDMI)	+= exynos_hdmi.o
>  exynosdrm-$(CONFIG_DRM_EXYNOS_VIDI)	+= exynos_drm_vidi.o
>  exynosdrm-$(CONFIG_DRM_EXYNOS_G2D)	+= exynos_drm_g2d.o
> +exynosdrm-$(CONFIG_DRM_EXYNOS_IPP)	+= exynos_drm_ipp.o
>  exynosdrm-$(CONFIG_DRM_EXYNOS_FIMC)	+= exynos_drm_fimc.o
>  exynosdrm-$(CONFIG_DRM_EXYNOS_ROTATOR)	+= exynos_drm_rotator.o
>  exynosdrm-$(CONFIG_DRM_EXYNOS_GSC)	+= exynos_drm_gsc.o
> diff --git a/drivers/gpu/drm/exynos/exynos_drm_drv.c b/drivers/gpu/drm/exynos/exynos_drm_drv.c
> index 7f19054ebce3..645046c5943a 100644
> --- a/drivers/gpu/drm/exynos/exynos_drm_drv.c
> +++ b/drivers/gpu/drm/exynos/exynos_drm_drv.c
> @@ -26,6 +26,7 @@
>  #include "exynos_drm_fb.h"
>  #include "exynos_drm_gem.h"
>  #include "exynos_drm_plane.h"
> +#include "exynos_drm_ipp.h"
>  #include "exynos_drm_vidi.h"
>  #include "exynos_drm_g2d.h"
>  #include "exynos_drm_iommu.h"
> @@ -114,6 +115,14 @@ static void exynos_drm_lastclose(struct drm_device *dev)
>  			DRM_AUTH | DRM_RENDER_ALLOW),
>  	DRM_IOCTL_DEF_DRV(EXYNOS_G2D_EXEC, exynos_g2d_exec_ioctl,
>  			DRM_AUTH | DRM_RENDER_ALLOW),
> +	DRM_IOCTL_DEF_DRV(EXYNOS_IPP_GET_RESOURCES, exynos_drm_ipp_get_res_ioctl,
> +			DRM_AUTH | DRM_RENDER_ALLOW),
> +	DRM_IOCTL_DEF_DRV(EXYNOS_IPP_GET_CAPS, exynos_drm_ipp_get_caps_ioctl,
> +			DRM_AUTH | DRM_RENDER_ALLOW),
> +	DRM_IOCTL_DEF_DRV(EXYNOS_IPP_GET_LIMITS, exynos_drm_ipp_get_limits_ioctl,
> +			DRM_AUTH | DRM_RENDER_ALLOW),
> +	DRM_IOCTL_DEF_DRV(EXYNOS_IPP_COMMIT, exynos_drm_ipp_commit_ioctl,
> +			DRM_AUTH | DRM_RENDER_ALLOW),
>  };
>  
>  static const struct file_operations exynos_drm_driver_fops = {
> diff --git a/drivers/gpu/drm/exynos/exynos_drm_ipp.c b/drivers/gpu/drm/exynos/exynos_drm_ipp.c
> new file mode 100644
> index 000000000000..11be6a1bf839
> --- /dev/null
> +++ b/drivers/gpu/drm/exynos/exynos_drm_ipp.c
> @@ -0,0 +1,892 @@
> +/*
> + * Copyright (C) 2017 Samsung Electronics Co.Ltd
> + * Authors:
> + *	Marek Szyprowski <m.szyprowski at samsung.com>
> + *
> + * Exynos DRM Image Post Processing (IPP) related functions
> + *
> + * Permission is hereby granted, free of charge, to any person obtaining a
> + * copy of this software and associated documentation files (the "Software"),
> + * to deal in the Software without restriction, including without limitation
> + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
> + * and/or sell copies of the Software, and to permit persons to whom the
> + * Software is furnished to do so, subject to the following conditions:
> + *
> + * The above copyright notice and this permission notice shall be included in
> + * all copies or substantial portions of the Software.
> + */
> +
> +
> +#include <drm/drmP.h>
> +#include <drm/drm_mode.h>
> +#include <uapi/drm/exynos_drm.h>
> +
> +#include "exynos_drm_drv.h"
> +#include "exynos_drm_gem.h"
> +#include "exynos_drm_ipp.h"
> +
> +static int num_ipp;
> +static LIST_HEAD(ipp_list);
> +
> +struct drm_pending_exynos_ipp_event {
> +	struct drm_pending_event base;
> +	struct drm_exynos_ipp_event event;
> +};
> +
> +/**
> + * exynos_drm_ipp_register - Register a new picture processor hardware module
> + * @dev: DRM device
> + * @ipp: ipp module to init
> + * @funcs: callbacks for the new ipp object
> + * @caps: bitmask of ipp capabilities (%DRM_EXYNOS_IPP_CAP_*)
> + * @formats: array of supported formats (%DRM_FORMAT_*)
> + * @name: name (for debugging purposes)
> + *
> + * Initializes a ipp module.
> + *
> + * Returns:
> + * Zero on success, error code on failure.
> + */
> +int exynos_drm_ipp_register(struct drm_device *dev, struct exynos_drm_ipp *ipp,
> +		const struct exynos_drm_ipp_funcs *funcs, unsigned int caps,
> +		const struct exynos_drm_ipp_formats *formats, const char *name)
> +{
> +	WARN_ON(!ipp);
> +	WARN_ON(!funcs);
> +	WARN_ON(!formats);
> +
> +	spin_lock_init(&ipp->lock);
> +	INIT_LIST_HEAD(&ipp->todo_list);
> +	init_waitqueue_head(&ipp->done_wq);
> +	ipp->dev = dev;
> +	ipp->funcs = funcs;
> +	ipp->capabilities = caps;
> +	ipp->name = name;
> +	ipp->formats = formats;
> +	while (formats++->fourcc)
> +		ipp->num_formats++;
> +
> +	list_add_tail(&ipp->head, &ipp_list);
> +	ipp->id = num_ipp++;
> +
> +	DRM_DEBUG_DRIVER("Registered ipp %d\n", ipp->id);
> +
> +	return 0;
> +}
> +
> +/**
> + * exynos_drm_ipp_unregister - Unregister the picture processor module
> + * @dev: DRM device
> + * @ipp: ipp module
> + */
> +void exynos_drm_ipp_unregister(struct drm_device *dev,
> +			       struct exynos_drm_ipp *ipp)
> +{
> +	BUG_ON(ipp->task);
> +	BUG_ON(!list_empty(&ipp->todo_list));
> +	list_del(&ipp->head);
> +}
> +
> +/**
> + * exynos_drm_ipp_ioctl_get_res_ioctl - enumerate all ipp modules
> + * @dev: DRM device
> + * @data: ioctl data
> + * @file_priv: DRM file info
> + *
> + * Construct a list of ipp ids.
> + *
> + * Called by the user via ioctl.
> + *
> + * Returns:
> + * Zero on success, negative errno on failure.
> + */
> +int exynos_drm_ipp_get_res_ioctl(struct drm_device *dev, void *data,
> +				 struct drm_file *file_priv)
> +{
> +	struct drm_exynos_ioctl_ipp_get_res *resp = data;
> +	struct exynos_drm_ipp *ipp;
> +	uint32_t __user *ipp_ptr = (uint32_t __user *)
> +						(unsigned long)resp->ipp_id_ptr;
> +	unsigned int count = num_ipp, copied = 0;
> +
> +	/*
> +	 * This ioctl is called twice, once to determine how much space is
> +	 * needed, and the 2nd time to fill it.
> +	 */
> +	if (count && resp->count_ipps >= count) {
> +		list_for_each_entry(ipp, &ipp_list, head) {
> +			if (put_user(ipp->id, ipp_ptr + copied))
> +				return -EFAULT;
> +			copied++;
> +		}
> +	}
> +	resp->count_ipps = count;
> +
> +	return 0;
> +}
> +
> +static inline struct exynos_drm_ipp *__ipp_get(uint32_t id)
> +{
> +	struct exynos_drm_ipp *ipp;
> +
> +	list_for_each_entry(ipp, &ipp_list, head)
> +		if (ipp->id == id)
> +			return ipp;
> +	return NULL;
> +}
> +
> +/**
> + * exynos_drm_ipp_ioctl_get_caps - get ipp module capabilities and formats
> + * @dev: DRM device
> + * @data: ioctl data
> + * @file_priv: DRM file info
> + *
> + * Construct a structure describing ipp module capabilities.
> + *
> + * Called by the user via ioctl.
> + *
> + * Returns:
> + * Zero on success, negative errno on failure.
> + */
> +int exynos_drm_ipp_get_caps_ioctl(struct drm_device *dev, void *data,
> +				  struct drm_file *file_priv)
> +{
> +	struct drm_exynos_ioctl_ipp_get_caps *resp = data;
> +	void __user *ptr = (void __user *)(unsigned long)resp->formats_ptr;
> +	struct exynos_drm_ipp *ipp;
> +	int i;
> +
> +	ipp = __ipp_get(resp->ipp_id);
> +	if (!ipp)
> +		return -ENOENT;
> +
> +	resp->ipp_id = ipp->id;
> +	resp->capabilities = ipp->capabilities;
> +
> +	/*
> +	 * This ioctl is called twice, once to determine how much space is
> +	 * needed, and the 2nd time to fill it.
> +	 */
> +	if (resp->formats_count >= ipp->num_formats) {
> +		for (i = 0; i < ipp->num_formats; i++) {
> +			struct exynos_drm_ipp_format tmp = {
> +				.fourcc = ipp->formats[i].fourcc,
> +				.type = ipp->formats[i].type,
> +				.modifier = ipp->formats[i].modifier,
> +			};
> +
> +			if (copy_to_user(ptr, &tmp, sizeof(tmp)))
> +				return -EFAULT;
> +			ptr += sizeof(tmp);
> +		}
> +	}
> +	resp->formats_count = ipp->num_formats;
> +
> +	return 0;
> +}
> +
> +static inline const struct exynos_drm_ipp_formats *__ipp_format_get(
> +			uint32_t fourcc, uint64_t mod, unsigned long type,
> +			const struct exynos_drm_ipp_formats *f)
> +{
> +	for (; f->fourcc; f++)
> +		if ((f->type & type) && f->fourcc == fourcc &&
> +		    f->modifier == mod)
> +			return f;
> +	return NULL;
> +}
> +
> +/**
> + * exynos_drm_ipp_get_limits_ioctl - get ipp module limits
> + * @dev: DRM device
> + * @data: ioctl data
> + * @file_priv: DRM file info
> + *
> + * Construct a structure describing ipp module limitations for provided
> + * picture format.
> + *
> + * Called by the user via ioctl.
> + *
> + * Returns:
> + * Zero on success, negative errno on failure.
> + */
> +int exynos_drm_ipp_get_limits_ioctl(struct drm_device *dev, void *data,
> +				    struct drm_file *file_priv)
I'm getting a null-pointer Opps in this ioctl(). I haven't fully debugged this,
but I guess it's because format->limits might be NULL. E.g. the formats for the
FIMC (exynos_drm_ipp_formats fimc_formats[]) don't carry any limits for the
color formats.




> +{
> +	struct drm_exynos_ioctl_ipp_get_limits *resp = data;
> +	void __user *ptr = (void __user *)(unsigned long)resp->limits_ptr;
> +	const struct exynos_drm_ipp_formats *format;
> +	const struct drm_exynos_ipp_limit *limits, *l;
> +	struct exynos_drm_ipp *ipp;
> +	int count = 0;
> +
> +	if (resp->type != DRM_EXYNOS_IPP_FORMAT_SOURCE &&
> +	    resp->type != DRM_EXYNOS_IPP_FORMAT_DESTINATION)
> +		return -EINVAL;
> +
> +	ipp = __ipp_get(resp->ipp_id);
> +	if (!ipp)
> +		return -ENOENT;
> +
> +	format = __ipp_format_get(resp->fourcc, resp->modifier, resp->type,
> +			      ipp->formats);
> +	if (!format)
> +		return -EINVAL;
> +
> +	limits = format->limits;
> +	for (l = limits; l->type; l++)
> +		count++;
> +
> +	/*
> +	 * This ioctl is called twice, once to determine how much space is
> +	 * needed, and the 2nd time to fill it.
> +	 */
> +	if (count && resp->limits_size >= count * sizeof(*l)) {
> +		for (l = limits; l->type; l++, ptr += sizeof(*l))
> +			if (copy_to_user((void __user *)ptr, l, sizeof(*l)))
> +				return -EFAULT;
> +	}
> +	resp->limits_size = count * sizeof(*l);
I wonder why you return the size if bytes here? How about keeping this
consistant with the other ioctls (like exynos_drm_ipp_get_caps_ioctl) and just
return the count (together with renaming limits_size to limits>count).

- Tobias


> +
> +	return 0;
> +}
> +
> +static inline struct exynos_drm_ipp_task *
> +	exynos_drm_ipp_task_alloc(struct exynos_drm_ipp *ipp)
> +{
> +	struct exynos_drm_ipp_task *task;
> +
> +	task = kzalloc(sizeof(*task), GFP_KERNEL);
> +	if (!task)
> +		return NULL;
> +
> +	task->dev = ipp->dev;
> +	task->ipp = ipp;
> +
> +	/* some defaults */
> +	task->src.rect.w = task->dst.rect.w = UINT_MAX;
> +	task->src.rect.h = task->dst.rect.h = UINT_MAX;
> +	task->transform.rotation = DRM_MODE_ROTATE_0;
> +
> +	DRM_DEBUG_DRIVER("Allocated task %pK\n", task);
> +
> +	return task;
> +}
> +
> +struct exynos_drm_param_map {
> +	unsigned int id;
> +	unsigned int size;
> +	unsigned int offset;
> +} exynos_drm_ipp_params_maps[] = {
> +	{
> +		DRM_EXYNOS_IPP_TASK_BUFFER | DRM_EXYNOS_IPP_TASK_TYPE_SOURCE,
> +		sizeof(struct drm_exynos_ipp_task_buffer),
> +		offsetof(struct exynos_drm_ipp_task, src.buf),
> +	}, {
> +		DRM_EXYNOS_IPP_TASK_BUFFER | DRM_EXYNOS_IPP_TASK_TYPE_DESTINATION,
> +		sizeof(struct drm_exynos_ipp_task_buffer),
> +		offsetof(struct exynos_drm_ipp_task, dst.buf),
> +	}, {
> +		DRM_EXYNOS_IPP_TASK_RECTANGLE | DRM_EXYNOS_IPP_TASK_TYPE_SOURCE,
> +		sizeof(struct drm_exynos_ipp_task_rect),
> +		offsetof(struct exynos_drm_ipp_task, src.rect),
> +	}, {
> +		DRM_EXYNOS_IPP_TASK_RECTANGLE | DRM_EXYNOS_IPP_TASK_TYPE_DESTINATION,
> +		sizeof(struct drm_exynos_ipp_task_rect),
> +		offsetof(struct exynos_drm_ipp_task, dst.rect),
> +	}, {
> +		DRM_EXYNOS_IPP_TASK_TRANSFORM,
> +		sizeof(struct drm_exynos_ipp_task_transform),
> +		offsetof(struct exynos_drm_ipp_task, transform),
> +	}, {
> +		DRM_EXYNOS_IPP_TASK_ALPHA,
> +		sizeof(struct drm_exynos_ipp_task_alpha),
> +		offsetof(struct exynos_drm_ipp_task, alpha),
> +	},
> +};
> +
> +static int exynos_drm_ipp_task_set(struct exynos_drm_ipp_task *task,
> +				  struct drm_exynos_ioctl_ipp_commit *arg)
> +{
> +	unsigned int size = arg->params_size;
> +	void *p, *params;
> +	int ret = 0;
> +
> +	if (size > PAGE_SIZE)
> +		return -ERANGE;
> +
> +	p = kmalloc(size, GFP_KERNEL);
> +	if (!p)
> +		return -ENOMEM;
> +
> +	if (copy_from_user(p, (void __user *)(unsigned long)arg->params_ptr,
> +			   size)) {
> +		ret = -EFAULT;
> +		DRM_DEBUG_DRIVER("Failed to copy configuration from userspace\n");
> +		goto free;
> +	}
> +
> +	params = p;
> +	while (size) {
> +		struct exynos_drm_param_map *map = exynos_drm_ipp_params_maps;
> +		u32 *id = params;
> +		int i;
> +
> +		for (i = 0; i < ARRAY_SIZE(exynos_drm_ipp_params_maps); i++)
> +			if (map[i].id == *id)
> +				break;
> +
> +		if (i < ARRAY_SIZE(exynos_drm_ipp_params_maps) &&
> +		    map[i].size <= size) {
> +			memcpy((void *)task + map[i].offset, params,
> +			       map[i].size);
> +			params += map[i].size;
> +			size -= map[i].size;
> +		} else {
> +			ret = -EINVAL;
> +			goto free;
> +		}
> +	}
> +
> +	DRM_DEBUG_DRIVER("Got task %pK configuration from userspace\n", task);
> +free:
> +	kfree(p);
> +	return ret;
> +}
> +
> +static int exynos_drm_ipp_task_setup_buffer(struct exynos_drm_ipp_buffer *buf,
> +					    struct drm_file *filp)
> +{
> +	int ret = 0;
> +	int i;
> +
> +	/* basic checks */
> +	if (buf->buf.width == 0 || buf->buf.height == 0)
> +		return -EINVAL;
> +	buf->format = drm_format_info(buf->buf.fourcc);
> +	for (i = 0; i < buf->format->num_planes; i++) {
> +		unsigned int width = (i == 0) ? buf->buf.width :
> +			     DIV_ROUND_UP(buf->buf.width, buf->format->hsub);
> +
> +		if (buf->buf.pitch[i] == 0)
> +			buf->buf.pitch[i] = width * buf->format->cpp[i];
> +		if (buf->buf.pitch[i] < width * buf->format->cpp[i])
> +			return -EINVAL;
> +		if (!buf->buf.gem_id[i])
> +			return -ENOENT;
> +	}
> +
> +	/* get GEM buffers and check their size */
> +	for (i = 0; i < buf->format->num_planes; i++) {
> +		unsigned int height = (i == 0) ? buf->buf.height :
> +			     DIV_ROUND_UP(buf->buf.height, buf->format->vsub);
> +		unsigned long size = height * buf->buf.pitch[i] +
> +				     buf->buf.offset[i];
> +		struct drm_gem_object *obj = drm_gem_object_lookup(filp,
> +							    buf->buf.gem_id[i]);
> +		if (!obj) {
> +			ret = -ENOENT;
> +			goto gem_free;
> +		}
> +		buf->exynos_gem[i] = to_exynos_gem(obj);
> +
> +		if (size > buf->exynos_gem[i]->size) {
> +			i++;
> +			ret = -EINVAL;
> +			goto gem_free;
> +		}
> +		buf->dma_addr[i] = buf->exynos_gem[i]->dma_addr +
> +				   buf->buf.offset[i];
> +	}
> +
> +	return 0;
> +gem_free:
> +	while (i--) {
> +		drm_gem_object_put_unlocked(&buf->exynos_gem[i]->base);
> +		buf->exynos_gem[i] = NULL;
> +	}
> +	return ret;
> +}
> +
> +static void exynos_drm_ipp_task_release_buffer(struct exynos_drm_ipp_buffer *buf)
> +{
> +	int i;
> +
> +	if (!buf->exynos_gem[0])
> +		return;
> +	for (i = 0; i < buf->format->num_planes; i++)
> +		drm_gem_object_put_unlocked(&buf->exynos_gem[i]->base);
> +}
> +
> +static void exynos_drm_ipp_task_free(struct exynos_drm_ipp *ipp,
> +				 struct exynos_drm_ipp_task *task)
> +{
> +	DRM_DEBUG_DRIVER("Freeing task %pK\n", task);
> +
> +	exynos_drm_ipp_task_release_buffer(&task->src);
> +	exynos_drm_ipp_task_release_buffer(&task->dst);
> +	if (task->event)
> +		drm_event_cancel_free(ipp->dev, &task->event->base);
> +	kfree(task);
> +}
> +
> +struct drm_ipp_limit {
> +	struct drm_exynos_ipp_limit_val h;
> +	struct drm_exynos_ipp_limit_val v;
> +};
> +
> +enum drm_ipp_size_id {
> +	IPP_LIMIT_BUFFER, IPP_LIMIT_AREA, IPP_LIMIT_ROTATED, IPP_LIMIT_MAX
> +};
> +
> +static const enum drm_ipp_size_id limit_id_fallback[IPP_LIMIT_MAX][4] = {
> +	[IPP_LIMIT_BUFFER]  = { DRM_EXYNOS_IPP_LIMIT_SIZE_BUFFER },
> +	[IPP_LIMIT_AREA]    = { DRM_EXYNOS_IPP_LIMIT_SIZE_AREA,
> +				DRM_EXYNOS_IPP_LIMIT_SIZE_BUFFER },
> +	[IPP_LIMIT_ROTATED] = { DRM_EXYNOS_IPP_LIMIT_SIZE_ROTATED,
> +				DRM_EXYNOS_IPP_LIMIT_SIZE_AREA,
> +				DRM_EXYNOS_IPP_LIMIT_SIZE_BUFFER },
> +};
> +
> +static inline void __limit_set_val(unsigned int *ptr, unsigned int val)
> +{
> +	if (!*ptr)
> +		*ptr = val;
> +}
> +
> +static void __get_size_limit(const struct drm_exynos_ipp_limit *limits,
> +			     enum drm_ipp_size_id id, struct drm_ipp_limit *res)
> +{
> +	const struct drm_exynos_ipp_limit *l = limits;
> +	int i = 0;
> +
> +	memset(res, 0, sizeof(*res));
> +	for (i = 0; limit_id_fallback[id][i]; i++)
> +		for (l = limits; l->type; l++) {
> +			if (((l->type & DRM_EXYNOS_IPP_LIMIT_TYPE_MASK) !=
> +			      DRM_EXYNOS_IPP_LIMIT_TYPE_SIZE) ||
> +			    ((l->type & DRM_EXYNOS_IPP_LIMIT_SIZE_MASK) !=
> +						     limit_id_fallback[id][i]))
> +				continue;
> +			__limit_set_val(&res->h.min, l->h.min);
> +			__limit_set_val(&res->h.max, l->h.max);
> +			__limit_set_val(&res->h.align, l->h.align);
> +			__limit_set_val(&res->v.min, l->v.min);
> +			__limit_set_val(&res->v.max, l->v.max);
> +			__limit_set_val(&res->v.align, l->v.align);
> +		}
> +}
> +
> +static inline bool __align_check(unsigned int val, unsigned int align)
> +{
> +	if (align && (val & (align - 1)))
> +		return false;
> +	return true;
> +}
> +
> +static inline bool __size_limit_check(unsigned int val,
> +				 struct drm_exynos_ipp_limit_val *l)
> +{
> +	if ((l->min && val < l->min) || (l->max && val > l->max) ||
> +	    !__align_check(val, l->align))
> +		return false;
> +	return true;
> +}
> +
> +static int exynos_drm_ipp_check_size_limits(struct exynos_drm_ipp_buffer *buf,
> +	const struct drm_exynos_ipp_limit *limits, bool rotate, bool swap)
> +{
> +	enum drm_ipp_size_id id = rotate ? IPP_LIMIT_ROTATED : IPP_LIMIT_AREA;
> +	struct drm_ipp_limit l;
> +	struct drm_exynos_ipp_limit_val *lh = &l.h, *lv = &l.v;
> +
> +	if (!limits)
> +		return 0;
> +
> +	__get_size_limit(limits, IPP_LIMIT_BUFFER, &l);
> +	if (!__size_limit_check(buf->buf.width, &l.h) ||
> +	    !__size_limit_check(buf->buf.height, &l.v))
> +		return -EINVAL;
> +
> +	if (swap) {
> +		lv = &l.h;
> +		lh = &l.v;
> +	}
> +	__get_size_limit(limits, id, &l);
> +	if (!__size_limit_check(buf->rect.w, lh) ||
> +	    !__align_check(buf->rect.x, lh->align) ||
> +	    !__size_limit_check(buf->rect.h, lv) ||
> +	    !__align_check(buf->rect.y, lv->align))
> +		return -EINVAL;
> +
> +	return 0;
> +}
> +
> +static int exynos_drm_ipp_check_scale_limits(struct drm_exynos_ipp_task_rect *src,
> +	struct drm_exynos_ipp_task_rect *dst,
> +	const struct drm_exynos_ipp_limit *limits, bool swap)
> +{
> +	const struct drm_exynos_ipp_limit_val *lh, *lv;
> +
> +	if (!limits)
> +		return 0;
> +
> +	for (; limits->type; limits++)
> +		if ((limits->type & DRM_EXYNOS_IPP_LIMIT_TYPE_MASK) ==
> +		    DRM_EXYNOS_IPP_LIMIT_TYPE_SCALE)
> +			break;
> +	if (!limits->type)
> +		return 0;
> +
> +	lh = (!swap) ? &limits->h : &limits->v;
> +	lv = (!swap) ? &limits->v : &limits->h;
> +
> +	if ((lh->max && dst->w > src->w * lh->max) ||
> +	    (lh->min && (dst->w << 16) < src->w * lh->min) ||
> +	    (lv->max && dst->h > src->h * lv->max) ||
> +	    (lv->min && (dst->h << 16) < src->h * lv->min))
> +		return -EINVAL;
> +
> +	return 0;
> +}
> +
> +static int exynos_drm_ipp_task_check(struct exynos_drm_ipp_task *task,
> +				    struct drm_file *filp)
> +{
> +	struct exynos_drm_ipp *ipp = task->ipp;
> +	const struct exynos_drm_ipp_formats *src_format, *dst_format;
> +	struct exynos_drm_ipp_buffer *src = &task->src, *dst = &task->dst;
> +	unsigned int rotation = task->transform.rotation;
> +	int ret = 0;
> +	bool scale = false, swap = false;
> +
> +	DRM_DEBUG_DRIVER("Checking %pK\n", task);
> +
> +	if (src->rect.w == UINT_MAX)
> +		src->rect.w = src->buf.width;
> +	if (src->rect.h == UINT_MAX)
> +		src->rect.h = src->buf.height;
> +	if (dst->rect.w == UINT_MAX)
> +		dst->rect.w = dst->buf.width;
> +	if (dst->rect.h == UINT_MAX)
> +		dst->rect.h = dst->buf.height;
> +
> +	if (src->rect.x + src->rect.w > (src->buf.width) ||
> +	    src->rect.y + src->rect.h > (src->buf.height) ||
> +	    dst->rect.x + dst->rect.w > (dst->buf.width) ||
> +	    dst->rect.y + dst->rect.h > (dst->buf.height))
> +		return -EINVAL;
> +
> +	DRM_DEBUG_DRIVER("Task %pK: basic checks done\n", task);
> +
> +	if (!(ipp->capabilities & DRM_EXYNOS_IPP_CAP_CROP) &&
> +	    (src->rect.x || src->rect.y || dst->rect.x || dst->rect.y))
> +		return -EINVAL;
> +
> +	if (!(ipp->capabilities & DRM_EXYNOS_IPP_CAP_ROTATE) &&
> +	    rotation != DRM_MODE_ROTATE_0)
> +		return -EINVAL;
> +
> +	if (drm_rotation_90_or_270(rotation))
> +		swap = true;
> +
> +	if ((!swap && (src->rect.w != dst->rect.w || src->rect.h != dst->rect.h)) ||
> +	    (swap && (src->rect.w != dst->rect.h || src->rect.h != dst->rect.w)))
> +		scale = true;
> +
> +	if (!(ipp->capabilities & DRM_EXYNOS_IPP_CAP_SCALE) && scale)
> +		return -EINVAL;
> +
> +	if (!(ipp->capabilities & DRM_EXYNOS_IPP_CAP_CONVERT) &&
> +	    src->buf.fourcc != dst->buf.fourcc)
> +		return -EINVAL;
> +
> +	DRM_DEBUG_DRIVER("Task %pK: capability checks done\n", task);
> +
> +	src_format = __ipp_format_get(src->buf.fourcc, src->buf.modifier,
> +				  DRM_EXYNOS_IPP_FORMAT_SOURCE, ipp->formats);
> +	if (!src_format)
> +		return -EINVAL;
> +	ret = exynos_drm_ipp_check_size_limits(src, src_format->limits,
> +					  rotation != DRM_MODE_ROTATE_0, false);
> +	if (ret)
> +		return ret;
> +	ret = exynos_drm_ipp_check_scale_limits(&src->rect, &dst->rect,
> +						src_format->limits, swap);
> +	if (ret)
> +		return ret;
> +
> +	DRM_DEBUG_DRIVER("Task %pK: source image checks done\n", task);
> +
> +	dst_format = __ipp_format_get(dst->buf.fourcc, dst->buf.modifier,
> +			       DRM_EXYNOS_IPP_FORMAT_DESTINATION, ipp->formats);
> +	if (!dst_format)
> +		return -EINVAL;
> +	ret = exynos_drm_ipp_check_size_limits(dst, dst_format->limits,
> +					   rotation != DRM_MODE_ROTATE_0, swap);
> +	if (ret)
> +		return ret;
> +	ret = exynos_drm_ipp_check_scale_limits(&src->rect, &dst->rect,
> +						dst_format->limits, swap);
> +	if (ret)
> +		return ret;
> +
> +	DRM_DEBUG_DRIVER("Task %pK: destination image checks done\n", task);
> +
> +	ret = exynos_drm_ipp_task_setup_buffer(src, filp);
> +	if (ret)
> +		return ret;
> +
> +	DRM_DEBUG_DRIVER("Task %pK: source buffer checks done\n", task);
> +
> +	ret = exynos_drm_ipp_task_setup_buffer(dst, filp);
> +	if (ret)
> +		return ret;
> +
> +	DRM_DEBUG_DRIVER("Task %pK: destination buffer checks done\n", task);
> +
> +	if (ipp->funcs->check)
> +		ret = ipp->funcs->check(ipp, task);
> +
> +	DRM_DEBUG_DRIVER("Task %pK: all checks done.\n", task);
> +
> +	return ret;
> +}
> +
> +static int exynos_drm_ipp_event_create(struct exynos_drm_ipp_task *task,
> +			struct drm_file *file_priv, uint64_t user_data)
> +{
> +	struct drm_pending_exynos_ipp_event *e = NULL;
> +	int ret;
> +
> +	e = kzalloc(sizeof(*e), GFP_KERNEL);
> +	if (!e)
> +		return -ENOMEM;
> +
> +	e->event.base.type = DRM_EXYNOS_IPP_EVENT;
> +	e->event.base.length = sizeof(e->event);
> +	e->event.user_data = user_data;
> +
> +	ret = drm_event_reserve_init(task->dev, file_priv, &e->base,
> +				     &e->event.base);
> +	if (ret)
> +		goto free;
> +
> +	task->event = e;
> +	return 0;
> +free:
> +	kfree(e);
> +	return ret;
> +}
> +
> +static void exynos_drm_ipp_event_send(struct exynos_drm_ipp_task *task)
> +{
> +	struct timeval now = ktime_to_timeval(ktime_get());
> +
> +	task->event->event.tv_sec = now.tv_sec;
> +	task->event->event.tv_usec = now.tv_usec;
> +	task->event->event.sequence = atomic_inc_return(&task->ipp->sequence);
> +
> +	drm_send_event(task->dev, &task->event->base);
> +}
> +
> +static int exynos_drm_ipp_task_cleanup(struct exynos_drm_ipp_task *task)
> +{
> +	int ret = task->ret;
> +
> +	if (ret == 0 && task->event) {
> +		exynos_drm_ipp_event_send(task);
> +		/* ensure event won't be canceled on task free */
> +		task->event = NULL;
> +	}
> +
> +	exynos_drm_ipp_task_free(task->ipp, task);
> +	return ret;
> +}
> +
> +static void exynos_drm_ipp_cleanup_work(struct work_struct *work)
> +{
> +	struct exynos_drm_ipp_task *task = container_of(work,
> +				struct exynos_drm_ipp_task, cleanup_work);
> +
> +	exynos_drm_ipp_task_cleanup(task);
> +}
> +
> +static void exynos_drm_ipp_next_task(struct exynos_drm_ipp *ipp);
> +
> +/**
> + * exynos_drm_ipp_task_done - finish given task and set return code
> + * @task: ipp task to finish
> + * @ret: error code or 0 if operation has been performed successfully
> + */
> +void exynos_drm_ipp_task_done(struct exynos_drm_ipp_task *task, int ret)
> +{
> +	struct exynos_drm_ipp *ipp = task->ipp;
> +	unsigned long flags;
> +
> +	DRM_DEBUG_DRIVER("ipp: %d, task %pK done\n", ipp->id, task);
> +
> +	spin_lock_irqsave(&ipp->lock, flags);
> +	if (ipp->task == task)
> +		ipp->task = NULL;
> +	task->flags |= DRM_EXYNOS_IPP_TASK_DONE;
> +	task->ret = ret;
> +	spin_unlock_irqrestore(&ipp->lock, flags);
> +
> +	exynos_drm_ipp_next_task(ipp);
> +	wake_up(&ipp->done_wq);
> +
> +	if (task->flags & DRM_EXYNOS_IPP_TASK_ASYNC) {
> +		INIT_WORK(&task->cleanup_work, exynos_drm_ipp_cleanup_work);
> +		schedule_work(&task->cleanup_work);
> +	}
> +}
> +
> +static void exynos_drm_ipp_next_task(struct exynos_drm_ipp *ipp)
> +{
> +	struct exynos_drm_ipp_task *task;
> +	unsigned long flags;
> +	int ret;
> +
> +	DRM_DEBUG_DRIVER("ipp: %d, try to run new task\n", ipp->id);
> +
> +	spin_lock_irqsave(&ipp->lock, flags);
> +
> +	if (ipp->task || list_empty(&ipp->todo_list)) {
> +		spin_unlock_irqrestore(&ipp->lock, flags);
> +		return;
> +	}
> +
> +	task = list_first_entry(&ipp->todo_list, struct exynos_drm_ipp_task,
> +				head);
> +	list_del_init(&task->head);
> +	ipp->task = task;
> +
> +	spin_unlock_irqrestore(&ipp->lock, flags);
> +
> +	DRM_DEBUG_DRIVER("ipp: %d, selected task %pK to run\n", ipp->id, task);
> +
> +	ret = ipp->funcs->commit(ipp, task);
> +	if (ret)
> +		exynos_drm_ipp_task_done(task, ret);
> +}
> +
> +static void exynos_drm_ipp_schedule_task(struct exynos_drm_ipp *ipp,
> +				     struct exynos_drm_ipp_task *task)
> +{
> +	unsigned long flags;
> +
> +	spin_lock_irqsave(&ipp->lock, flags);
> +	list_add(&task->head, &ipp->todo_list);
> +	spin_unlock_irqrestore(&ipp->lock, flags);
> +
> +	exynos_drm_ipp_next_task(ipp);
> +}
> +
> +static void exynos_drm_ipp_task_abort(struct exynos_drm_ipp *ipp,
> +				  struct exynos_drm_ipp_task *task)
> +{
> +	unsigned long flags;
> +
> +	spin_lock_irqsave(&ipp->lock, flags);
> +	if (task->flags & DRM_EXYNOS_IPP_TASK_DONE) {
> +		/* already completed task */
> +		exynos_drm_ipp_task_cleanup(task);
> +	} else if (ipp->task != task) {
> +		/* task has not been scheduled for execution yet */
> +		list_del_init(&task->head);
> +		exynos_drm_ipp_task_cleanup(task);
> +	} else {
> +		/*
> +		 * currently processed task, call abort() and perform
> +		 * cleanup with async worker
> +		 */
> +		task->flags |= DRM_EXYNOS_IPP_TASK_ASYNC;
> +		spin_unlock_irqrestore(&ipp->lock, flags);
> +		if (ipp->funcs->abort)
> +			ipp->funcs->abort(ipp, task);
> +		return;
> +	}
> +	spin_unlock_irqrestore(&ipp->lock, flags);
> +}
> +
> +/**
> + * exynos_drm_ipp_commit_ioctl - perform image processing operation
> + * @dev: DRM device
> + * @data: ioctl data
> + * @file_priv: DRM file info
> + *
> + * Construct a ipp task from the set of properties provided from the user
> + * and try to schedule it to framebuffer processor hardware.
> + *
> + * Called by the user via ioctl.
> + *
> + * Returns:
> + * Zero on success, negative errno on failure.
> + */
> +int exynos_drm_ipp_commit_ioctl(struct drm_device *dev, void *data,
> +			  struct drm_file *file_priv)
> +{
> +	struct drm_exynos_ioctl_ipp_commit *arg = data;
> +	struct exynos_drm_ipp *ipp;
> +	struct exynos_drm_ipp_task *task;
> +	int ret = 0;
> +
> +	if ((arg->flags & ~DRM_EXYNOS_IPP_FLAGS) || arg->reserved)
> +		return -EINVAL;
> +
> +	/* can't test and expect an event at the same time */
> +	if ((arg->flags & DRM_EXYNOS_IPP_FLAG_TEST_ONLY) &&
> +			(arg->flags & DRM_EXYNOS_IPP_FLAG_EVENT))
> +		return -EINVAL;
> +
> +	ipp = __ipp_get(arg->ipp_id);
> +	if (!ipp)
> +		return -ENOENT;
> +
> +	task = exynos_drm_ipp_task_alloc(ipp);
> +	if (!task)
> +		return -ENOMEM;
> +
> +	ret = exynos_drm_ipp_task_set(task, arg);
> +	if (ret)
> +		goto free;
> +
> +	ret = exynos_drm_ipp_task_check(task, file_priv);
> +	if (ret || arg->flags & DRM_EXYNOS_IPP_FLAG_TEST_ONLY)
> +		goto free;
> +
> +	if (arg->flags & DRM_EXYNOS_IPP_FLAG_EVENT) {
> +		ret = exynos_drm_ipp_event_create(task, file_priv,
> +						 arg->user_data);
> +		if (ret)
> +			goto free;
> +	}
> +
> +	/*
> +	 * Queue task for processing on the hardware. task object will be
> +	 * then freed after exynos_drm_ipp_task_done()
> +	 */
> +	if (arg->flags & DRM_EXYNOS_IPP_FLAG_NONBLOCK) {
> +		DRM_DEBUG_DRIVER("ipp: %d, nonblocking processing task %pK\n",
> +				 ipp->id, task);
> +
> +		task->flags |= DRM_EXYNOS_IPP_TASK_ASYNC;
> +		exynos_drm_ipp_schedule_task(task->ipp, task);
> +		ret = 0;
> +	} else {
> +		DRM_DEBUG_DRIVER("ipp: %d, processing task %pK\n", ipp->id, task);
> +		exynos_drm_ipp_schedule_task(ipp, task);
> +		ret = wait_event_interruptible(ipp->done_wq,
> +					task->flags & DRM_EXYNOS_IPP_TASK_DONE);
> +		if (ret)
> +			exynos_drm_ipp_task_abort(ipp, task);
> +		else
> +			ret = exynos_drm_ipp_task_cleanup(task);
> +	}
> +	return ret;
> +free:
> +	exynos_drm_ipp_task_free(ipp, task);
> +
> +	return ret;
> +}
> diff --git a/drivers/gpu/drm/exynos/exynos_drm_ipp.h b/drivers/gpu/drm/exynos/exynos_drm_ipp.h
> new file mode 100644
> index 000000000000..532a60698a6a
> --- /dev/null
> +++ b/drivers/gpu/drm/exynos/exynos_drm_ipp.h
> @@ -0,0 +1,188 @@
> +/*
> + * Copyright (c) 2017 Samsung Electronics Co., Ltd.
> + *
> + * 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 _EXYNOS_DRM_IPP_H_
> +#define _EXYNOS_DRM_IPP_H_
> +
> +#include <drm/drmP.h>
> +
> +struct exynos_drm_ipp;
> +struct exynos_drm_ipp_task;
> +
> +/**
> + * struct exynos_drm_ipp_funcs - exynos_drm_ipp control functions
> + */
> +struct exynos_drm_ipp_funcs {
> +	/**
> +	 * @check:
> +	 *
> +	 * This is the optional hook to validate an ipp task. This function
> +	 * must reject any task which the hardware or driver doesn't support.
> +	 * This includes but is of course not limited to:
> +	 *
> +	 *  - Checking that the buffers, scaling and placement requirements
> +	 *    and so on are within the limits of the hardware not specified by
> +	 *    limits array.
> +	 *
> +	 *  - The driver does not need to repeat basic input validation like
> +	 *    done in the exynos_drm_ipp_task_check() function. The core does
> +	 *    that before calling this hook.
> +	 *
> +	 * RETURNS:
> +	 *
> +	 * 0 on success or one of the below negative error codes:
> +	 *
> +	 *  - -EINVAL, if any of the above constraints are violated.
> +	 */
> +	int (*check)(struct exynos_drm_ipp *ipp,
> +		     struct exynos_drm_ipp_task *task);
> +
> +	/**
> +	 * @commit:
> +	 *
> +	 * This is the main entry point to start framebuffer processing
> +	 * in the hardware. The exynos_drm_ipp_task has been already validated.
> +	 * This function must not wait until the device finishes processing.
> +	 * When the driver finishes processing, it has to call
> +	 * exynos_exynos_drm_ipp_task_done() function.
> +	 *
> +	 * RETURNS:
> +	 *
> +	 * 0 on success or negative error codes in case of failure.
> +	 */
> +	int (*commit)(struct exynos_drm_ipp *ipp,
> +		      struct exynos_drm_ipp_task *task);
> +
> +	/**
> +	 * @abort:
> +	 *
> +	 * Informs the driver that it has to abort the currently running
> +	 * task as soon as possible (i.e. as soon as it can stop the device
> +	 * safely), even if the task would not have been finished by then.
> +	 * After the driver performs the necessary steps, it has to call
> +	 * exynos_drm_ipp_task_done() (as if the task ended normally).
> +	 * This function does not have to (and will usually not) wait
> +	 * until the device enters a state when it can be stopped.
> +	 */
> +	void (*abort)(struct exynos_drm_ipp *ipp,
> +		      struct exynos_drm_ipp_task *task);
> +};
> +
> +/**
> + * struct exynos_drm_ipp - central picture processor module structure
> + */
> +struct exynos_drm_ipp {
> +	struct drm_device *dev;
> +	struct list_head head;
> +	unsigned int id;
> +
> +	const char *name;
> +	const struct exynos_drm_ipp_funcs *funcs;
> +	unsigned int capabilities;
> +	const struct exynos_drm_ipp_formats *formats;
> +	unsigned int num_formats;
> +	atomic_t sequence;
> +
> +	spinlock_t lock;
> +	struct exynos_drm_ipp_task *task;
> +	struct list_head todo_list;
> +	wait_queue_head_t done_wq;
> +};
> +
> +struct exynos_drm_ipp_buffer {
> +	struct drm_exynos_ipp_task_buffer buf;
> +	struct drm_exynos_ipp_task_rect rect;
> +
> +	struct exynos_drm_gem *exynos_gem[MAX_FB_BUFFER];
> +	const struct drm_format_info *format;
> +	dma_addr_t dma_addr[MAX_FB_BUFFER];
> +};
> +
> +/**
> + * struct exynos_drm_ipp_task - a structure describing transformation that
> + * has to be performed by the picture processor hardware module
> + */
> +struct exynos_drm_ipp_task {
> +	struct drm_device *dev;
> +	struct exynos_drm_ipp *ipp;
> +	struct list_head head;
> +
> +	struct exynos_drm_ipp_buffer src;
> +	struct exynos_drm_ipp_buffer dst;
> +
> +	struct drm_exynos_ipp_task_transform transform;
> +	struct drm_exynos_ipp_task_alpha alpha;
> +
> +	struct work_struct cleanup_work;
> +	unsigned int flags;
> +	int ret;
> +
> +	struct drm_pending_exynos_ipp_event *event;
> +};
> +
> +#define DRM_EXYNOS_IPP_TASK_DONE	(1 << 0)
> +#define DRM_EXYNOS_IPP_TASK_ASYNC	(1 << 1)
> +
> +struct exynos_drm_ipp_formats {
> +	u32 fourcc;
> +	u32 type;
> +	u64 modifier;
> +	const struct drm_exynos_ipp_limit *limits;
> +};
> +
> +/* helper macros to set exynos_drm_ipp_formats structure and limits*/
> +#define IPP_SRCDST_MFORMAT(f, m, l) \
> +     .fourcc = DRM_FORMAT_##f, .modifier = m, .limits = l, \
> +     .type = (DRM_EXYNOS_IPP_FORMAT_SOURCE | DRM_EXYNOS_IPP_FORMAT_DESTINATION)
> +
> +#define IPP_SRCDST_FORMAT(f, l) IPP_SRCDST_MFORMAT(f, 0, l)
> +
> +#define IPP_SIZE_LIMIT(l, val...)	\
> +     .type = (DRM_EXYNOS_IPP_LIMIT_TYPE_SIZE | DRM_EXYNOS_IPP_LIMIT_SIZE_##l), \
> +     val
> +
> +int exynos_drm_ipp_register(struct drm_device *dev, struct exynos_drm_ipp *ipp,
> +		const struct exynos_drm_ipp_funcs *funcs, unsigned int caps,
> +		const struct exynos_drm_ipp_formats *formats, const char *name);
> +void exynos_drm_ipp_unregister(struct drm_device *dev, struct exynos_drm_ipp *ipp);
> +
> +void exynos_drm_ipp_task_done(struct exynos_drm_ipp_task *task, int ret);
> +
> +#ifdef CONFIG_DRM_EXYNOS_IPP
> +int exynos_drm_ipp_get_res_ioctl(struct drm_device *dev, void *data,
> +				 struct drm_file *file_priv);
> +int exynos_drm_ipp_get_caps_ioctl(struct drm_device *dev, void *data,
> +				  struct drm_file *file_priv);
> +int exynos_drm_ipp_get_limits_ioctl(struct drm_device *dev, void *data,
> +				    struct drm_file *file_priv);
> +int exynos_drm_ipp_commit_ioctl(struct drm_device *dev,
> +				void *data, struct drm_file *file_priv);
> +#else
> +static inline int exynos_drm_ipp_get_res_ioctl(struct drm_device *dev,
> +	 void *data, struct drm_file *file_priv)
> +{
> +	return -ENODEV;
> +}
> +static inline int exynos_drm_ipp_get_caps_ioctl(struct drm_device *dev,
> +	 void *data, struct drm_file *file_priv)
> +{
> +	return -ENODEV;
> +}
> +static inline int exynos_drm_ipp_get_limits_ioctl(struct drm_device *dev,
> +	 void *data, struct drm_file *file_priv)
> +{
> +	return -ENODEV;
> +}
> +static inline int exynos_drm_ipp_commit_ioctl(struct drm_device *dev,
> +	 void *data, struct drm_file *file_priv)
> +{
> +	return -ENODEV;
> +}
> +#endif
> +#endif
> diff --git a/include/uapi/drm/exynos_drm.h b/include/uapi/drm/exynos_drm.h
> index 544db2cecb36..239e09022d75 100644
> --- a/include/uapi/drm/exynos_drm.h
> +++ b/include/uapi/drm/exynos_drm.h
> @@ -134,6 +134,208 @@ struct drm_exynos_g2d_exec {
>  	__u64					async;
>  };
>  
> +/* Exynos DRM IPP v2 API */
> +
> +/**
> + * Enumerate available IPP hardware modules.
> + *
> + * @count_ipps: size of ipp_id array / number of ipp modules (set by driver)
> + * @reserved: padding
> + * @ipp_id_ptr: pointer to ipp_id array or NULL
> + */
> +struct drm_exynos_ioctl_ipp_get_res {
> +	__u32 count_ipps;
> +	__u32 reserved;
> +	__u64 ipp_id_ptr;
> +};
> +
> +enum drm_exynos_ipp_format_type {
> +	DRM_EXYNOS_IPP_FORMAT_SOURCE		= 0x01,
> +	DRM_EXYNOS_IPP_FORMAT_DESTINATION	= 0x02,
> +};
> +
> +struct exynos_drm_ipp_format {
> +	__u32 fourcc;
> +	__u32 type;
> +	__u64 modifier;
> +};
> +
> +enum drm_exynos_ipp_capability {
> +	DRM_EXYNOS_IPP_CAP_CROP		= 0x01,
> +	DRM_EXYNOS_IPP_CAP_ROTATE	= 0x02,
> +	DRM_EXYNOS_IPP_CAP_SCALE	= 0x04,
> +	DRM_EXYNOS_IPP_CAP_CONVERT	= 0x08,
> +};
> +
> +/**
> + * Get IPP hardware capabilities and supported image formats.
> + *
> + * @ipp_id: id of IPP module to query
> + * @capabilities: bitmask of drm_exynos_ipp_capability (set by driver)
> + * @reserved: padding
> + * @formats_count: size of formats array (in entries) / number of filled
> + *		   formats (set by driver)
> + * @formats_ptr: pointer to formats array or NULL
> + */
> +struct drm_exynos_ioctl_ipp_get_caps {
> +	__u32 ipp_id;
> +	__u32 capabilities;
> +	__u32 reserved;
> +	__u32 formats_count;
> +	__u64 formats_ptr;
> +};
> +
> +enum drm_exynos_ipp_limit_type {
> +	/* size (horizontal/vertial) limits, in pixels (min, max, alignment) */
> +	DRM_EXYNOS_IPP_LIMIT_TYPE_SIZE		= 0x0001,
> +	/* scale ratio (horizonta/vertial), 16.16 fixed point (min, max) */
> +	DRM_EXYNOS_IPP_LIMIT_TYPE_SCALE		= 0x0002,
> +
> +	/* image buffer area */
> +	DRM_EXYNOS_IPP_LIMIT_SIZE_BUFFER	= 0x0001 << 16,
> +	/* src/dst rectangle area */
> +	DRM_EXYNOS_IPP_LIMIT_SIZE_AREA		= 0x0002 << 16,
> +	/* src/dst rectangle area when rotation enabled */
> +	DRM_EXYNOS_IPP_LIMIT_SIZE_ROTATED	= 0x0003 << 16,
> +
> +	DRM_EXYNOS_IPP_LIMIT_TYPE_MASK		= 0x000f,
> +	DRM_EXYNOS_IPP_LIMIT_SIZE_MASK		= 0x000f << 16,
> +};
> +
> +struct drm_exynos_ipp_limit_val {
> +	__u32 min, max, align;
> +};
> +
> +/**
> + * IPP module limitation.
> + *
> + * @type: limit type (see drm_exynos_ipp_limit_type enum)
> + * @h, @v: horizontal and vertical limits
> + */
> +struct drm_exynos_ipp_limit {
> +	__u32 type;
> +	struct drm_exynos_ipp_limit_val h;
> +	struct drm_exynos_ipp_limit_val v;
> +};
> +
> +/**
> + * Get IPP limits for given image format.
> + *
> + * @ipp_id: id of IPP module to query
> + * @fourcc: image format code (see DRM_FORMAT_* in drm_fourcc.h)
> + * @modifier: image format modifier (see DRM_FORMAT_MOD_* in drm_fourcc.h)
> + * @type: source/destination identifier (drm_exynos_ipp_format_flag enum)
> + * @limits_size: size of limits array (in bytes) / size of filled entries
> + *		 (set by driver)
> + * @limits_ptr: pointer to limits array or NULL
> + */
> +struct drm_exynos_ioctl_ipp_get_limits {
> +	__u32 ipp_id;
> +	__u32 fourcc;
> +	__u64 modifier;
> +	__u32 type;
> +	__u32 limits_size;
> +	__u64 limits_ptr;
> +};
> +
> +enum drm_exynos_ipp_task_id {
> +	/* buffer described by struct drm_exynos_ipp_task_buffer */
> +	DRM_EXYNOS_IPP_TASK_BUFFER		= 0x0001,
> +	/* rectangle described by struct drm_exynos_ipp_task_rect */
> +	DRM_EXYNOS_IPP_TASK_RECTANGLE		= 0x0002,
> +	/* transformation described by struct drm_exynos_ipp_task_transform */
> +	DRM_EXYNOS_IPP_TASK_TRANSFORM		= 0x0003,
> +	/* alpha configuration described by struct drm_exynos_ipp_task_alpha */
> +	DRM_EXYNOS_IPP_TASK_ALPHA		= 0x0004,
> +
> +	/* source image data (for buffer and rectangle chunks) */
> +	DRM_EXYNOS_IPP_TASK_TYPE_SOURCE		= 0x0001 << 16,
> +	/* destination image data (for buffer and rectangle chunks) */
> +	DRM_EXYNOS_IPP_TASK_TYPE_DESTINATION	= 0x0002 << 16,
> +};
> +
> +/**
> + * Memory buffer with image data.
> + *
> + * @id: must be DRM_EXYNOS_IPP_TASK_BUFFER
> + * other parameters are same as for AddFB2 generic DRM ioctl
> + */
> +struct drm_exynos_ipp_task_buffer {
> +	__u32	id;
> +	__u32	fourcc;
> +	__u32	width, height;
> +	__u32	gem_id[4];
> +	__u32	offset[4];
> +	__u32	pitch[4];
> +	__u64	modifier;
> +};
> +
> +/**
> + * Rectangle for processing.
> + *
> + * @id: must be DRM_EXYNOS_IPP_TASK_RECTANGLE
> + * @x, at y: left corner in pixels
> + * @w, at h: width/height in pixels
> + */
> +struct drm_exynos_ipp_task_rect {
> +	__u32	id;
> +	__u32	x, y, w, h;
> +};
> +
> +/**
> + * Image tranformation description.
> + *
> + * @id: must be DRM_EXYNOS_IPP_TASK_TRANSFORM
> + * @rotation: DRM_MODE_ROTATE_* and DRM_MODE_REFLECT_* values
> + */
> +struct drm_exynos_ipp_task_transform {
> +	__u32	id;
> +	__u32	rotation;
> +};
> +
> +/**
> + * Image global alpha configuration for formats without alpha values.
> + *
> + * @id: must be DRM_EXYNOS_IPP_TASK_ALPHA
> + * @value: global alpha value (0-255)
> + */
> +struct drm_exynos_ipp_task_alpha {
> +	__u32	id;
> +	__u32	value;
> +};
> +
> +enum drm_exynos_ipp_flag {
> +	/* generate DRM event after processing */
> +	DRM_EXYNOS_IPP_FLAG_EVENT	= 0x01,
> +	/* dry run, only check task parameters */
> +	DRM_EXYNOS_IPP_FLAG_TEST_ONLY	= 0x02,
> +	/* non-blocking processing */
> +	DRM_EXYNOS_IPP_FLAG_NONBLOCK	= 0x04,
> +};
> +
> +#define DRM_EXYNOS_IPP_FLAGS (DRM_EXYNOS_IPP_FLAG_EVENT |\
> +		DRM_EXYNOS_IPP_FLAG_TEST_ONLY | DRM_EXYNOS_IPP_FLAG_NONBLOCK)
> +
> +/**
> + * Perform image processing described by array of drm_exynos_ipp_task_*
> + * structures (parameters array).
> + *
> + * @ipp_id: id of IPP module to run the task
> + * @flags: bitmask of drm_exynos_ipp_flag values
> + * @reserved: padding
> + * @params_size: size of parameters array (in bytes)
> + * @params_ptr: pointer to parameters array or NULL
> + * @user_data: (optional) data for drm event
> + */
> +struct drm_exynos_ioctl_ipp_commit {
> +	__u32 ipp_id;
> +	__u32 flags;
> +	__u32 reserved;
> +	__u32 params_size;
> +	__u64 params_ptr;
> +	__u64 user_data;
> +};
> +
>  #define DRM_EXYNOS_GEM_CREATE		0x00
>  #define DRM_EXYNOS_GEM_MAP		0x01
>  /* Reserved 0x03 ~ 0x05 for exynos specific gem ioctl */
> @@ -146,6 +348,11 @@ struct drm_exynos_g2d_exec {
>  #define DRM_EXYNOS_G2D_EXEC		0x22
>  
>  /* Reserved 0x30 ~ 0x33 for obsolete Exynos IPP ioctls */
> +/* IPP - Image Post Processing */
> +#define DRM_EXYNOS_IPP_GET_RESOURCES	0x40
> +#define DRM_EXYNOS_IPP_GET_CAPS		0x41
> +#define DRM_EXYNOS_IPP_GET_LIMITS	0x42
> +#define DRM_EXYNOS_IPP_COMMIT		0x43
>  
>  #define DRM_IOCTL_EXYNOS_GEM_CREATE		DRM_IOWR(DRM_COMMAND_BASE + \
>  		DRM_EXYNOS_GEM_CREATE, struct drm_exynos_gem_create)
> @@ -164,8 +371,18 @@ struct drm_exynos_g2d_exec {
>  #define DRM_IOCTL_EXYNOS_G2D_EXEC		DRM_IOWR(DRM_COMMAND_BASE + \
>  		DRM_EXYNOS_G2D_EXEC, struct drm_exynos_g2d_exec)
>  
> +#define DRM_IOCTL_EXYNOS_IPP_GET_RESOURCES	DRM_IOWR(DRM_COMMAND_BASE + \
> +		DRM_EXYNOS_IPP_GET_RESOURCES, struct drm_exynos_ioctl_ipp_get_res)
> +#define DRM_IOCTL_EXYNOS_IPP_GET_CAPS		DRM_IOWR(DRM_COMMAND_BASE + \
> +		DRM_EXYNOS_IPP_GET_CAPS, struct drm_exynos_ioctl_ipp_get_caps)
> +#define DRM_IOCTL_EXYNOS_IPP_GET_LIMITS		DRM_IOWR(DRM_COMMAND_BASE + \
> +		DRM_EXYNOS_IPP_GET_LIMITS, struct drm_exynos_ioctl_ipp_get_limits)
> +#define DRM_IOCTL_EXYNOS_IPP_COMMIT		DRM_IOWR(DRM_COMMAND_BASE + \
> +		DRM_EXYNOS_IPP_COMMIT, struct drm_exynos_ioctl_ipp_commit)
> +
>  /* EXYNOS specific events */
>  #define DRM_EXYNOS_G2D_EVENT		0x80000000
> +#define DRM_EXYNOS_IPP_EVENT		0x80000002
>  
>  struct drm_exynos_g2d_event {
>  	struct drm_event	base;
> @@ -176,6 +393,16 @@ struct drm_exynos_g2d_event {
>  	__u32			reserved;
>  };
>  
> +struct drm_exynos_ipp_event {
> +	struct drm_event	base;
> +	__u64			user_data;
> +	__u32			tv_sec;
> +	__u32			tv_usec;
> +	__u32			ipp_id;
> +	__u32			sequence;
> +	__u64			reserved;
> +};
> +
>  #if defined(__cplusplus)
>  }
>  #endif
> 



More information about the dri-devel mailing list