diff options
Diffstat (limited to 'drivers/media/platform/vsp1/vsp1_rwpf.c')
| -rw-r--r-- | drivers/media/platform/vsp1/vsp1_rwpf.c | 220 | 
1 files changed, 220 insertions, 0 deletions
diff --git a/drivers/media/platform/vsp1/vsp1_rwpf.c b/drivers/media/platform/vsp1/vsp1_rwpf.c new file mode 100644 index 00000000000..ec3dab6a9b9 --- /dev/null +++ b/drivers/media/platform/vsp1/vsp1_rwpf.c @@ -0,0 +1,220 @@ +/* + * vsp1_rwpf.c  --  R-Car VSP1 Read and Write Pixel Formatters + * + * Copyright (C) 2013-2014 Renesas Electronics Corporation + * + * Contact: Laurent Pinchart (laurent.pinchart@ideasonboard.com) + * + * 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. + */ + +#include <media/v4l2-subdev.h> + +#include "vsp1.h" +#include "vsp1_rwpf.h" +#include "vsp1_video.h" + +#define RWPF_MIN_WIDTH				1 +#define RWPF_MIN_HEIGHT				1 + +/* ----------------------------------------------------------------------------- + * V4L2 Subdevice Pad Operations + */ + +int vsp1_rwpf_enum_mbus_code(struct v4l2_subdev *subdev, +			     struct v4l2_subdev_fh *fh, +			     struct v4l2_subdev_mbus_code_enum *code) +{ +	static const unsigned int codes[] = { +		V4L2_MBUS_FMT_ARGB8888_1X32, +		V4L2_MBUS_FMT_AYUV8_1X32, +	}; + +	if (code->index >= ARRAY_SIZE(codes)) +		return -EINVAL; + +	code->code = codes[code->index]; + +	return 0; +} + +int vsp1_rwpf_enum_frame_size(struct v4l2_subdev *subdev, +			      struct v4l2_subdev_fh *fh, +			      struct v4l2_subdev_frame_size_enum *fse) +{ +	struct vsp1_rwpf *rwpf = to_rwpf(subdev); +	struct v4l2_mbus_framefmt *format; + +	format = v4l2_subdev_get_try_format(fh, fse->pad); + +	if (fse->index || fse->code != format->code) +		return -EINVAL; + +	if (fse->pad == RWPF_PAD_SINK) { +		fse->min_width = RWPF_MIN_WIDTH; +		fse->max_width = rwpf->max_width; +		fse->min_height = RWPF_MIN_HEIGHT; +		fse->max_height = rwpf->max_height; +	} else { +		/* The size on the source pad are fixed and always identical to +		 * the size on the sink pad. +		 */ +		fse->min_width = format->width; +		fse->max_width = format->width; +		fse->min_height = format->height; +		fse->max_height = format->height; +	} + +	return 0; +} + +static struct v4l2_rect * +vsp1_rwpf_get_crop(struct vsp1_rwpf *rwpf, struct v4l2_subdev_fh *fh, u32 which) +{ +	switch (which) { +	case V4L2_SUBDEV_FORMAT_TRY: +		return v4l2_subdev_get_try_crop(fh, RWPF_PAD_SINK); +	case V4L2_SUBDEV_FORMAT_ACTIVE: +		return &rwpf->crop; +	default: +		return NULL; +	} +} + +int vsp1_rwpf_get_format(struct v4l2_subdev *subdev, struct v4l2_subdev_fh *fh, +			 struct v4l2_subdev_format *fmt) +{ +	struct vsp1_rwpf *rwpf = to_rwpf(subdev); + +	fmt->format = *vsp1_entity_get_pad_format(&rwpf->entity, fh, fmt->pad, +						  fmt->which); + +	return 0; +} + +int vsp1_rwpf_set_format(struct v4l2_subdev *subdev, struct v4l2_subdev_fh *fh, +			 struct v4l2_subdev_format *fmt) +{ +	struct vsp1_rwpf *rwpf = to_rwpf(subdev); +	struct v4l2_mbus_framefmt *format; +	struct v4l2_rect *crop; + +	/* Default to YUV if the requested format is not supported. */ +	if (fmt->format.code != V4L2_MBUS_FMT_ARGB8888_1X32 && +	    fmt->format.code != V4L2_MBUS_FMT_AYUV8_1X32) +		fmt->format.code = V4L2_MBUS_FMT_AYUV8_1X32; + +	format = vsp1_entity_get_pad_format(&rwpf->entity, fh, fmt->pad, +					    fmt->which); + +	if (fmt->pad == RWPF_PAD_SOURCE) { +		/* The RWPF performs format conversion but can't scale, only the +		 * format code can be changed on the source pad. +		 */ +		format->code = fmt->format.code; +		fmt->format = *format; +		return 0; +	} + +	format->code = fmt->format.code; +	format->width = clamp_t(unsigned int, fmt->format.width, +				RWPF_MIN_WIDTH, rwpf->max_width); +	format->height = clamp_t(unsigned int, fmt->format.height, +				 RWPF_MIN_HEIGHT, rwpf->max_height); +	format->field = V4L2_FIELD_NONE; +	format->colorspace = V4L2_COLORSPACE_SRGB; + +	fmt->format = *format; + +	/* Update the sink crop rectangle. */ +	crop = vsp1_rwpf_get_crop(rwpf, fh, fmt->which); +	crop->left = 0; +	crop->top = 0; +	crop->width = fmt->format.width; +	crop->height = fmt->format.height; + +	/* Propagate the format to the source pad. */ +	format = vsp1_entity_get_pad_format(&rwpf->entity, fh, RWPF_PAD_SOURCE, +					    fmt->which); +	*format = fmt->format; + +	return 0; +} + +int vsp1_rwpf_get_selection(struct v4l2_subdev *subdev, +			    struct v4l2_subdev_fh *fh, +			    struct v4l2_subdev_selection *sel) +{ +	struct vsp1_rwpf *rwpf = to_rwpf(subdev); +	struct v4l2_mbus_framefmt *format; + +	/* Cropping is implemented on the sink pad. */ +	if (sel->pad != RWPF_PAD_SINK) +		return -EINVAL; + +	switch (sel->target) { +	case V4L2_SEL_TGT_CROP: +		sel->r = *vsp1_rwpf_get_crop(rwpf, fh, sel->which); +		break; + +	case V4L2_SEL_TGT_CROP_BOUNDS: +		format = vsp1_entity_get_pad_format(&rwpf->entity, fh, +						    RWPF_PAD_SINK, sel->which); +		sel->r.left = 0; +		sel->r.top = 0; +		sel->r.width = format->width; +		sel->r.height = format->height; +		break; + +	default: +		return -EINVAL; +	} + +	return 0; +} + +int vsp1_rwpf_set_selection(struct v4l2_subdev *subdev, +			    struct v4l2_subdev_fh *fh, +			    struct v4l2_subdev_selection *sel) +{ +	struct vsp1_rwpf *rwpf = to_rwpf(subdev); +	struct v4l2_mbus_framefmt *format; +	struct v4l2_rect *crop; + +	/* Cropping is implemented on the sink pad. */ +	if (sel->pad != RWPF_PAD_SINK) +		return -EINVAL; + +	if (sel->target != V4L2_SEL_TGT_CROP) +		return -EINVAL; + +	/* Make sure the crop rectangle is entirely contained in the image. The +	 * WPF top and left offsets are limited to 255. +	 */ +	format = vsp1_entity_get_pad_format(&rwpf->entity, fh, RWPF_PAD_SINK, +					    sel->which); +	sel->r.left = min_t(unsigned int, sel->r.left, format->width - 2); +	sel->r.top = min_t(unsigned int, sel->r.top, format->height - 2); +	if (rwpf->entity.type == VSP1_ENTITY_WPF) { +		sel->r.left = min_t(unsigned int, sel->r.left, 255); +		sel->r.top = min_t(unsigned int, sel->r.top, 255); +	} +	sel->r.width = min_t(unsigned int, sel->r.width, +			     format->width - sel->r.left); +	sel->r.height = min_t(unsigned int, sel->r.height, +			      format->height - sel->r.top); + +	crop = vsp1_rwpf_get_crop(rwpf, fh, sel->which); +	*crop = sel->r; + +	/* Propagate the format to the source pad. */ +	format = vsp1_entity_get_pad_format(&rwpf->entity, fh, RWPF_PAD_SOURCE, +					    sel->which); +	format->width = crop->width; +	format->height = crop->height; + +	return 0; +}  | 
