mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 10:40:15 +02:00 
			
		
		
		
	drm/omap: Enable COLOR_ENCODING and COLOR_RANGE properties for planes
Adds support for COLOR_ENCODING and COLOR_RANGE properties to omap_plane.c and dispc.c. The supported encodings and ranges are presets are: For COLOR_ENCODING: - YCbCr BT.601 (default) - YCbCr BT.709 For COLOR_RANGE: - YCbCr limited range - YCbCr full range (default) Signed-off-by: Jyri Sarha <jsarha@ti.com> Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com> Reviewed-by: Laurent Pinchart <laurent.pinchart@ideasonboard.com> Link: https://patchwork.freedesktop.org/patch/msgid/20201103080310.164453-6-tomi.valkeinen@ti.com
This commit is contained in:
		
							parent
							
								
									1bea53df4d
								
							
						
					
					
						commit
						64ff189118
					
				
					 3 changed files with 97 additions and 46 deletions
				
			
		| 
						 | 
					@ -877,50 +877,62 @@ static void dispc_ovl_write_color_conv_coef(struct dispc_device *dispc,
 | 
				
			||||||
#undef CVAL
 | 
					#undef CVAL
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void dispc_wb_write_color_conv_coef(struct dispc_device *dispc,
 | 
					/* YUV -> RGB, ITU-R BT.601, full range */
 | 
				
			||||||
					   const struct csc_coef_rgb2yuv *ct)
 | 
					static const struct csc_coef_yuv2rgb coefs_yuv2rgb_bt601_full = {
 | 
				
			||||||
 | 
						256,   0,  358,		/* ry, rcb, rcr |1.000  0.000  1.402|*/
 | 
				
			||||||
 | 
						256, -88, -182,		/* gy, gcb, gcr |1.000 -0.344 -0.714|*/
 | 
				
			||||||
 | 
						256, 452,    0,		/* by, bcb, bcr |1.000  1.772  0.000|*/
 | 
				
			||||||
 | 
						true,			/* full range */
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* YUV -> RGB, ITU-R BT.601, limited range */
 | 
				
			||||||
 | 
					static const struct csc_coef_yuv2rgb coefs_yuv2rgb_bt601_lim = {
 | 
				
			||||||
 | 
						298,    0,  409,	/* ry, rcb, rcr |1.164  0.000  1.596|*/
 | 
				
			||||||
 | 
						298, -100, -208,	/* gy, gcb, gcr |1.164 -0.392 -0.813|*/
 | 
				
			||||||
 | 
						298,  516,    0,	/* by, bcb, bcr |1.164  2.017  0.000|*/
 | 
				
			||||||
 | 
						false,			/* limited range */
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* YUV -> RGB, ITU-R BT.709, full range */
 | 
				
			||||||
 | 
					static const struct csc_coef_yuv2rgb coefs_yuv2rgb_bt709_full = {
 | 
				
			||||||
 | 
						256,    0,  402,        /* ry, rcb, rcr |1.000  0.000  1.570|*/
 | 
				
			||||||
 | 
						256,  -48, -120,        /* gy, gcb, gcr |1.000 -0.187 -0.467|*/
 | 
				
			||||||
 | 
						256,  475,    0,        /* by, bcb, bcr |1.000  1.856  0.000|*/
 | 
				
			||||||
 | 
						true,                   /* full range */
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* YUV -> RGB, ITU-R BT.709, limited range */
 | 
				
			||||||
 | 
					static const struct csc_coef_yuv2rgb coefs_yuv2rgb_bt709_lim = {
 | 
				
			||||||
 | 
						298,    0,  459,	/* ry, rcb, rcr |1.164  0.000  1.793|*/
 | 
				
			||||||
 | 
						298,  -55, -136,	/* gy, gcb, gcr |1.164 -0.213 -0.533|*/
 | 
				
			||||||
 | 
						298,  541,    0,	/* by, bcb, bcr |1.164  2.112  0.000|*/
 | 
				
			||||||
 | 
						false,			/* limited range */
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void dispc_ovl_set_csc(struct dispc_device *dispc,
 | 
				
			||||||
 | 
								      enum omap_plane_id plane,
 | 
				
			||||||
 | 
								      enum drm_color_encoding color_encoding,
 | 
				
			||||||
 | 
								      enum drm_color_range color_range)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	const enum omap_plane_id plane = OMAP_DSS_WB;
 | 
						const struct csc_coef_yuv2rgb *csc;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define CVAL(x, y) (FLD_VAL(x, 26, 16) | FLD_VAL(y, 10, 0))
 | 
						switch (color_encoding) {
 | 
				
			||||||
 | 
						default:
 | 
				
			||||||
 | 
						case DRM_COLOR_YCBCR_BT601:
 | 
				
			||||||
 | 
							if (color_range == DRM_COLOR_YCBCR_FULL_RANGE)
 | 
				
			||||||
 | 
								csc = &coefs_yuv2rgb_bt601_full;
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
								csc = &coefs_yuv2rgb_bt601_lim;
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						case DRM_COLOR_YCBCR_BT709:
 | 
				
			||||||
 | 
							if (color_range == DRM_COLOR_YCBCR_FULL_RANGE)
 | 
				
			||||||
 | 
								csc = &coefs_yuv2rgb_bt709_full;
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
								csc = &coefs_yuv2rgb_bt709_lim;
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 0), CVAL(ct->yg,  ct->yr));
 | 
						dispc_ovl_write_color_conv_coef(dispc, plane, csc);
 | 
				
			||||||
	dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 1), CVAL(ct->crr, ct->yb));
 | 
					 | 
				
			||||||
	dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 2), CVAL(ct->crb, ct->crg));
 | 
					 | 
				
			||||||
	dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 3), CVAL(ct->cbg, ct->cbr));
 | 
					 | 
				
			||||||
	dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 4), CVAL(0, ct->cbb));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	REG_FLD_MOD(dispc, DISPC_OVL_ATTRIBUTES(plane), ct->full_range, 11, 11);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#undef CVAL
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static void dispc_setup_color_conv_coef(struct dispc_device *dispc)
 | 
					 | 
				
			||||||
{
 | 
					 | 
				
			||||||
	int i;
 | 
					 | 
				
			||||||
	int num_ovl = dispc_get_num_ovls(dispc);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	/* YUV -> RGB, ITU-R BT.601, limited range */
 | 
					 | 
				
			||||||
	const struct csc_coef_yuv2rgb coefs_yuv2rgb_bt601_lim = {
 | 
					 | 
				
			||||||
		298,    0,  409,	/* ry, rcb, rcr */
 | 
					 | 
				
			||||||
		298, -100, -208,	/* gy, gcb, gcr */
 | 
					 | 
				
			||||||
		298,  516,    0,	/* by, bcb, bcr */
 | 
					 | 
				
			||||||
		false,			/* limited range */
 | 
					 | 
				
			||||||
	};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	/* RGB -> YUV, ITU-R BT.601, limited range */
 | 
					 | 
				
			||||||
	const struct csc_coef_rgb2yuv coefs_rgb2yuv_bt601_lim = {
 | 
					 | 
				
			||||||
		 66, 129,  25,		/* yr,   yg,  yb */
 | 
					 | 
				
			||||||
		-38, -74, 112,		/* cbr, cbg, cbb */
 | 
					 | 
				
			||||||
		112, -94, -18,		/* crr, crg, crb */
 | 
					 | 
				
			||||||
		false,			/* limited range */
 | 
					 | 
				
			||||||
	};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	for (i = 1; i < num_ovl; i++)
 | 
					 | 
				
			||||||
		dispc_ovl_write_color_conv_coef(dispc, i, &coefs_yuv2rgb_bt601_lim);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (dispc->feat->has_writeback)
 | 
					 | 
				
			||||||
		dispc_wb_write_color_conv_coef(dispc, &coefs_rgb2yuv_bt601_lim);
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void dispc_ovl_set_ba0(struct dispc_device *dispc,
 | 
					static void dispc_ovl_set_ba0(struct dispc_device *dispc,
 | 
				
			||||||
| 
						 | 
					@ -2601,7 +2613,9 @@ static int dispc_ovl_setup_common(struct dispc_device *dispc,
 | 
				
			||||||
				  u8 pre_mult_alpha, u8 global_alpha,
 | 
									  u8 pre_mult_alpha, u8 global_alpha,
 | 
				
			||||||
				  enum omap_dss_rotation_type rotation_type,
 | 
									  enum omap_dss_rotation_type rotation_type,
 | 
				
			||||||
				  bool replication, const struct videomode *vm,
 | 
									  bool replication, const struct videomode *vm,
 | 
				
			||||||
				  bool mem_to_mem)
 | 
									  bool mem_to_mem,
 | 
				
			||||||
 | 
									  enum drm_color_encoding color_encoding,
 | 
				
			||||||
 | 
									  enum drm_color_range color_range)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	bool five_taps = true;
 | 
						bool five_taps = true;
 | 
				
			||||||
	bool fieldmode = false;
 | 
						bool fieldmode = false;
 | 
				
			||||||
| 
						 | 
					@ -2750,6 +2764,9 @@ static int dispc_ovl_setup_common(struct dispc_device *dispc,
 | 
				
			||||||
				      fieldmode, fourcc, rotation);
 | 
									      fieldmode, fourcc, rotation);
 | 
				
			||||||
		dispc_ovl_set_output_size(dispc, plane, out_width, out_height);
 | 
							dispc_ovl_set_output_size(dispc, plane, out_width, out_height);
 | 
				
			||||||
		dispc_ovl_set_vid_color_conv(dispc, plane, cconv);
 | 
							dispc_ovl_set_vid_color_conv(dispc, plane, cconv);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (plane != OMAP_DSS_WB)
 | 
				
			||||||
 | 
								dispc_ovl_set_csc(dispc, plane, color_encoding, color_range);
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	dispc_ovl_set_rotation_attrs(dispc, plane, rotation, rotation_type,
 | 
						dispc_ovl_set_rotation_attrs(dispc, plane, rotation, rotation_type,
 | 
				
			||||||
| 
						 | 
					@ -2786,7 +2803,8 @@ static int dispc_ovl_setup(struct dispc_device *dispc,
 | 
				
			||||||
		oi->screen_width, oi->pos_x, oi->pos_y, oi->width, oi->height,
 | 
							oi->screen_width, oi->pos_x, oi->pos_y, oi->width, oi->height,
 | 
				
			||||||
		oi->out_width, oi->out_height, oi->fourcc, oi->rotation,
 | 
							oi->out_width, oi->out_height, oi->fourcc, oi->rotation,
 | 
				
			||||||
		oi->zorder, oi->pre_mult_alpha, oi->global_alpha,
 | 
							oi->zorder, oi->pre_mult_alpha, oi->global_alpha,
 | 
				
			||||||
		oi->rotation_type, replication, vm, mem_to_mem);
 | 
							oi->rotation_type, replication, vm, mem_to_mem,
 | 
				
			||||||
 | 
							oi->color_encoding, oi->color_range);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return r;
 | 
						return r;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -2819,7 +2837,8 @@ static int dispc_wb_setup(struct dispc_device *dispc,
 | 
				
			||||||
		wi->buf_width, pos_x, pos_y, in_width, in_height, wi->width,
 | 
							wi->buf_width, pos_x, pos_y, in_width, in_height, wi->width,
 | 
				
			||||||
		wi->height, wi->fourcc, wi->rotation, zorder,
 | 
							wi->height, wi->fourcc, wi->rotation, zorder,
 | 
				
			||||||
		wi->pre_mult_alpha, global_alpha, wi->rotation_type,
 | 
							wi->pre_mult_alpha, global_alpha, wi->rotation_type,
 | 
				
			||||||
		replication, vm, mem_to_mem);
 | 
							replication, vm, mem_to_mem, DRM_COLOR_YCBCR_BT601,
 | 
				
			||||||
 | 
							DRM_COLOR_YCBCR_LIMITED_RANGE);
 | 
				
			||||||
	if (r)
 | 
						if (r)
 | 
				
			||||||
		return r;
 | 
							return r;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -3930,8 +3949,6 @@ static void _omap_dispc_initial_config(struct dispc_device *dispc)
 | 
				
			||||||
	    dispc->feat->has_gamma_table)
 | 
						    dispc->feat->has_gamma_table)
 | 
				
			||||||
		REG_FLD_MOD(dispc, DISPC_CONFIG, 1, 9, 9);
 | 
							REG_FLD_MOD(dispc, DISPC_CONFIG, 1, 9, 9);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	dispc_setup_color_conv_coef(dispc);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	dispc_set_loadmode(dispc, OMAP_DSS_LOAD_FRAME_ONLY);
 | 
						dispc_set_loadmode(dispc, OMAP_DSS_LOAD_FRAME_ONLY);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	dispc_init_fifos(dispc);
 | 
						dispc_init_fifos(dispc);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -7,6 +7,7 @@
 | 
				
			||||||
#ifndef __OMAP_DRM_DSS_H
 | 
					#ifndef __OMAP_DRM_DSS_H
 | 
				
			||||||
#define __OMAP_DRM_DSS_H
 | 
					#define __OMAP_DRM_DSS_H
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <drm/drm_color_mgmt.h>
 | 
				
			||||||
#include <drm/drm_crtc.h>
 | 
					#include <drm/drm_crtc.h>
 | 
				
			||||||
#include <drm/drm_mode.h>
 | 
					#include <drm/drm_mode.h>
 | 
				
			||||||
#include <linux/device.h>
 | 
					#include <linux/device.h>
 | 
				
			||||||
| 
						 | 
					@ -243,6 +244,9 @@ struct omap_overlay_info {
 | 
				
			||||||
	u8 global_alpha;
 | 
						u8 global_alpha;
 | 
				
			||||||
	u8 pre_mult_alpha;
 | 
						u8 pre_mult_alpha;
 | 
				
			||||||
	u8 zorder;
 | 
						u8 zorder;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						enum drm_color_encoding color_encoding;
 | 
				
			||||||
 | 
						enum drm_color_range color_range;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
struct omap_overlay_manager_info {
 | 
					struct omap_overlay_manager_info {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -59,6 +59,8 @@ static void omap_plane_atomic_update(struct drm_plane *plane,
 | 
				
			||||||
		info.pre_mult_alpha = 1;
 | 
							info.pre_mult_alpha = 1;
 | 
				
			||||||
	else
 | 
						else
 | 
				
			||||||
		info.pre_mult_alpha = 0;
 | 
							info.pre_mult_alpha = 0;
 | 
				
			||||||
 | 
						info.color_encoding = state->color_encoding;
 | 
				
			||||||
 | 
						info.color_range = state->color_range;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* update scanout: */
 | 
						/* update scanout: */
 | 
				
			||||||
	omap_framebuffer_update_scanout(state->fb, state, &info);
 | 
						omap_framebuffer_update_scanout(state->fb, state, &info);
 | 
				
			||||||
| 
						 | 
					@ -189,6 +191,8 @@ static void omap_plane_reset(struct drm_plane *plane)
 | 
				
			||||||
	 */
 | 
						 */
 | 
				
			||||||
	plane->state->zpos = plane->type == DRM_PLANE_TYPE_PRIMARY
 | 
						plane->state->zpos = plane->type == DRM_PLANE_TYPE_PRIMARY
 | 
				
			||||||
			   ? 0 : omap_plane->id;
 | 
								   ? 0 : omap_plane->id;
 | 
				
			||||||
 | 
						plane->state->color_encoding = DRM_COLOR_YCBCR_BT601;
 | 
				
			||||||
 | 
						plane->state->color_range = DRM_COLOR_YCBCR_FULL_RANGE;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static int omap_plane_atomic_set_property(struct drm_plane *plane,
 | 
					static int omap_plane_atomic_set_property(struct drm_plane *plane,
 | 
				
			||||||
| 
						 | 
					@ -232,6 +236,23 @@ static const struct drm_plane_funcs omap_plane_funcs = {
 | 
				
			||||||
	.atomic_get_property = omap_plane_atomic_get_property,
 | 
						.atomic_get_property = omap_plane_atomic_get_property,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static bool omap_plane_supports_yuv(struct drm_plane *plane)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						struct omap_drm_private *priv = plane->dev->dev_private;
 | 
				
			||||||
 | 
						struct omap_plane *omap_plane = to_omap_plane(plane);
 | 
				
			||||||
 | 
						const u32 *formats =
 | 
				
			||||||
 | 
							priv->dispc_ops->ovl_get_color_modes(priv->dispc, omap_plane->id);
 | 
				
			||||||
 | 
						u32 i;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (i = 0; formats[i]; i++)
 | 
				
			||||||
 | 
							if (formats[i] == DRM_FORMAT_YUYV ||
 | 
				
			||||||
 | 
							    formats[i] == DRM_FORMAT_UYVY ||
 | 
				
			||||||
 | 
							    formats[i] == DRM_FORMAT_NV12)
 | 
				
			||||||
 | 
								return true;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return false;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static const char *plane_id_to_name[] = {
 | 
					static const char *plane_id_to_name[] = {
 | 
				
			||||||
	[OMAP_DSS_GFX] = "gfx",
 | 
						[OMAP_DSS_GFX] = "gfx",
 | 
				
			||||||
	[OMAP_DSS_VIDEO1] = "vid1",
 | 
						[OMAP_DSS_VIDEO1] = "vid1",
 | 
				
			||||||
| 
						 | 
					@ -293,6 +314,15 @@ struct drm_plane *omap_plane_init(struct drm_device *dev,
 | 
				
			||||||
	drm_plane_create_blend_mode_property(plane, BIT(DRM_MODE_BLEND_PREMULTI) |
 | 
						drm_plane_create_blend_mode_property(plane, BIT(DRM_MODE_BLEND_PREMULTI) |
 | 
				
			||||||
					     BIT(DRM_MODE_BLEND_COVERAGE));
 | 
										     BIT(DRM_MODE_BLEND_COVERAGE));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (omap_plane_supports_yuv(plane))
 | 
				
			||||||
 | 
							drm_plane_create_color_properties(plane,
 | 
				
			||||||
 | 
											  BIT(DRM_COLOR_YCBCR_BT601) |
 | 
				
			||||||
 | 
											  BIT(DRM_COLOR_YCBCR_BT709),
 | 
				
			||||||
 | 
											  BIT(DRM_COLOR_YCBCR_FULL_RANGE) |
 | 
				
			||||||
 | 
											  BIT(DRM_COLOR_YCBCR_LIMITED_RANGE),
 | 
				
			||||||
 | 
											  DRM_COLOR_YCBCR_BT601,
 | 
				
			||||||
 | 
											  DRM_COLOR_YCBCR_FULL_RANGE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return plane;
 | 
						return plane;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
error:
 | 
					error:
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue