Commit 64ff1891 authored by Jyri Sarha's avatar Jyri Sarha Committed by Tomi Valkeinen

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: default avatarJyri Sarha <jsarha@ti.com>
Signed-off-by: default avatarTomi Valkeinen <tomi.valkeinen@ti.com>
Reviewed-by: default avatarLaurent Pinchart <laurent.pinchart@ideasonboard.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20201103080310.164453-6-tomi.valkeinen@ti.com
parent 1bea53df
...@@ -877,50 +877,62 @@ static void dispc_ovl_write_color_conv_coef(struct dispc_device *dispc, ...@@ -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|*/
const enum omap_plane_id plane = OMAP_DSS_WB; 256, -88, -182, /* gy, gcb, gcr |1.000 -0.344 -0.714|*/
256, 452, 0, /* by, bcb, bcr |1.000 1.772 0.000|*/
#define CVAL(x, y) (FLD_VAL(x, 26, 16) | FLD_VAL(y, 10, 0)) true, /* full range */
};
dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 0), CVAL(ct->yg, ct->yr)); /* YUV -> RGB, ITU-R BT.601, limited range */
dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 1), CVAL(ct->crr, ct->yb)); static const struct csc_coef_yuv2rgb coefs_yuv2rgb_bt601_lim = {
dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 2), CVAL(ct->crb, ct->crg)); 298, 0, 409, /* ry, rcb, rcr |1.164 0.000 1.596|*/
dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 3), CVAL(ct->cbg, ct->cbr)); 298, -100, -208, /* gy, gcb, gcr |1.164 -0.392 -0.813|*/
dispc_write_reg(dispc, DISPC_OVL_CONV_COEF(plane, 4), CVAL(0, ct->cbb)); 298, 516, 0, /* by, bcb, bcr |1.164 2.017 0.000|*/
false, /* limited range */
};
REG_FLD_MOD(dispc, DISPC_OVL_ATTRIBUTES(plane), ct->full_range, 11, 11); /* 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 */
};
#undef CVAL /* 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_setup_color_conv_coef(struct dispc_device *dispc) 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)
{ {
int i; const struct csc_coef_yuv2rgb *csc;
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 */ switch (color_encoding) {
const struct csc_coef_rgb2yuv coefs_rgb2yuv_bt601_lim = { default:
66, 129, 25, /* yr, yg, yb */ case DRM_COLOR_YCBCR_BT601:
-38, -74, 112, /* cbr, cbg, cbb */ if (color_range == DRM_COLOR_YCBCR_FULL_RANGE)
112, -94, -18, /* crr, crg, crb */ csc = &coefs_yuv2rgb_bt601_full;
false, /* limited range */ else
}; csc = &coefs_yuv2rgb_bt601_lim;
break;
for (i = 1; i < num_ovl; i++) case DRM_COLOR_YCBCR_BT709:
dispc_ovl_write_color_conv_coef(dispc, i, &coefs_yuv2rgb_bt601_lim); if (color_range == DRM_COLOR_YCBCR_FULL_RANGE)
csc = &coefs_yuv2rgb_bt709_full;
else
csc = &coefs_yuv2rgb_bt709_lim;
break;
}
if (dispc->feat->has_writeback) dispc_ovl_write_color_conv_coef(dispc, plane, csc);
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, ...@@ -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, ...@@ -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, ...@@ -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, ...@@ -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) ...@@ -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 @@ ...@@ -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 { ...@@ -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, ...@@ -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) ...@@ -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 = { ...@@ -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, ...@@ -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:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment