Commit fd2d2fc2 authored by Marek Szyprowski's avatar Marek Szyprowski Committed by Inki Dae

drm/exynos: introduce exynos_drm_plane_config structure

This patch adds common structure for keeping plane configuration and
capabilities data. This patch is inspired by similar code developed by
Tobias Jakobi.

Changelog v2:
- fix vidi_win_types(i) call. vidi_win_types is not a function.
Signed-off-by: default avatarMarek Szyprowski <m.szyprowski@samsung.com>
Signed-off-by: default avatarInki Dae <inki.dae@samsung.com>
parent ab144201
...@@ -26,7 +26,6 @@ ...@@ -26,7 +26,6 @@
#include "exynos_drm_iommu.h" #include "exynos_drm_iommu.h"
#define WINDOWS_NR 3 #define WINDOWS_NR 3
#define CURSOR_WIN 2
#define MIN_FB_WIDTH_FOR_16WORD_BURST 128 #define MIN_FB_WIDTH_FOR_16WORD_BURST 128
static const char * const decon_clks_name[] = { static const char * const decon_clks_name[] = {
...@@ -57,6 +56,7 @@ struct decon_context { ...@@ -57,6 +56,7 @@ struct decon_context {
struct drm_device *drm_dev; struct drm_device *drm_dev;
struct exynos_drm_crtc *crtc; struct exynos_drm_crtc *crtc;
struct exynos_drm_plane planes[WINDOWS_NR]; struct exynos_drm_plane planes[WINDOWS_NR];
struct exynos_drm_plane_config configs[WINDOWS_NR];
void __iomem *addr; void __iomem *addr;
struct clk *clks[ARRAY_SIZE(decon_clks_name)]; struct clk *clks[ARRAY_SIZE(decon_clks_name)];
int pipe; int pipe;
...@@ -72,6 +72,12 @@ static const uint32_t decon_formats[] = { ...@@ -72,6 +72,12 @@ static const uint32_t decon_formats[] = {
DRM_FORMAT_ARGB8888, DRM_FORMAT_ARGB8888,
}; };
static const enum drm_plane_type decon_win_types[WINDOWS_NR] = {
DRM_PLANE_TYPE_PRIMARY,
DRM_PLANE_TYPE_OVERLAY,
DRM_PLANE_TYPE_CURSOR,
};
static inline void decon_set_bits(struct decon_context *ctx, u32 reg, u32 mask, static inline void decon_set_bits(struct decon_context *ctx, u32 reg, u32 mask,
u32 val) u32 val)
{ {
...@@ -482,7 +488,6 @@ static int decon_bind(struct device *dev, struct device *master, void *data) ...@@ -482,7 +488,6 @@ static int decon_bind(struct device *dev, struct device *master, void *data)
struct exynos_drm_private *priv = drm_dev->dev_private; struct exynos_drm_private *priv = drm_dev->dev_private;
struct exynos_drm_plane *exynos_plane; struct exynos_drm_plane *exynos_plane;
enum exynos_drm_output_type out_type; enum exynos_drm_output_type out_type;
enum drm_plane_type type;
unsigned int win; unsigned int win;
int ret; int ret;
...@@ -492,10 +497,13 @@ static int decon_bind(struct device *dev, struct device *master, void *data) ...@@ -492,10 +497,13 @@ static int decon_bind(struct device *dev, struct device *master, void *data)
for (win = ctx->first_win; win < WINDOWS_NR; win++) { for (win = ctx->first_win; win < WINDOWS_NR; win++) {
int tmp = (win == ctx->first_win) ? 0 : win; int tmp = (win == ctx->first_win) ? 0 : win;
type = exynos_plane_get_type(tmp, CURSOR_WIN); ctx->configs[win].pixel_formats = decon_formats;
ctx->configs[win].num_pixel_formats = ARRAY_SIZE(decon_formats);
ctx->configs[win].zpos = win;
ctx->configs[win].type = decon_win_types[tmp];
ret = exynos_plane_init(drm_dev, &ctx->planes[win], ret = exynos_plane_init(drm_dev, &ctx->planes[win],
1 << ctx->pipe, type, decon_formats, 1 << ctx->pipe, &ctx->configs[win]);
ARRAY_SIZE(decon_formats), win);
if (ret) if (ret)
return ret; return ret;
} }
......
...@@ -41,13 +41,13 @@ ...@@ -41,13 +41,13 @@
#define MIN_FB_WIDTH_FOR_16WORD_BURST 128 #define MIN_FB_WIDTH_FOR_16WORD_BURST 128
#define WINDOWS_NR 2 #define WINDOWS_NR 2
#define CURSOR_WIN 1
struct decon_context { struct decon_context {
struct device *dev; struct device *dev;
struct drm_device *drm_dev; struct drm_device *drm_dev;
struct exynos_drm_crtc *crtc; struct exynos_drm_crtc *crtc;
struct exynos_drm_plane planes[WINDOWS_NR]; struct exynos_drm_plane planes[WINDOWS_NR];
struct exynos_drm_plane_config configs[WINDOWS_NR];
struct clk *pclk; struct clk *pclk;
struct clk *aclk; struct clk *aclk;
struct clk *eclk; struct clk *eclk;
...@@ -82,6 +82,11 @@ static const uint32_t decon_formats[] = { ...@@ -82,6 +82,11 @@ static const uint32_t decon_formats[] = {
DRM_FORMAT_BGRA8888, DRM_FORMAT_BGRA8888,
}; };
static const enum drm_plane_type decon_win_types[WINDOWS_NR] = {
DRM_PLANE_TYPE_PRIMARY,
DRM_PLANE_TYPE_CURSOR,
};
static void decon_wait_for_vblank(struct exynos_drm_crtc *crtc) static void decon_wait_for_vblank(struct exynos_drm_crtc *crtc)
{ {
struct decon_context *ctx = crtc->ctx; struct decon_context *ctx = crtc->ctx;
...@@ -637,8 +642,7 @@ static int decon_bind(struct device *dev, struct device *master, void *data) ...@@ -637,8 +642,7 @@ static int decon_bind(struct device *dev, struct device *master, void *data)
struct decon_context *ctx = dev_get_drvdata(dev); struct decon_context *ctx = dev_get_drvdata(dev);
struct drm_device *drm_dev = data; struct drm_device *drm_dev = data;
struct exynos_drm_plane *exynos_plane; struct exynos_drm_plane *exynos_plane;
enum drm_plane_type type; unsigned int i;
unsigned int zpos;
int ret; int ret;
ret = decon_ctx_initialize(ctx, drm_dev); ret = decon_ctx_initialize(ctx, drm_dev);
...@@ -647,11 +651,14 @@ static int decon_bind(struct device *dev, struct device *master, void *data) ...@@ -647,11 +651,14 @@ static int decon_bind(struct device *dev, struct device *master, void *data)
return ret; return ret;
} }
for (zpos = 0; zpos < WINDOWS_NR; zpos++) { for (i = 0; i < WINDOWS_NR; i++) {
type = exynos_plane_get_type(zpos, CURSOR_WIN); ctx->configs[i].pixel_formats = decon_formats;
ret = exynos_plane_init(drm_dev, &ctx->planes[zpos], ctx->configs[i].num_pixel_formats = ARRAY_SIZE(decon_formats);
1 << ctx->pipe, type, decon_formats, ctx->configs[i].zpos = i;
ARRAY_SIZE(decon_formats), zpos); ctx->configs[i].type = decon_win_types[i];
ret = exynos_plane_init(drm_dev, &ctx->planes[i],
1 << ctx->pipe, &ctx->configs[i]);
if (ret) if (ret)
return ret; return ret;
} }
......
...@@ -84,10 +84,29 @@ to_exynos_plane_state(struct drm_plane_state *state) ...@@ -84,10 +84,29 @@ to_exynos_plane_state(struct drm_plane_state *state)
struct exynos_drm_plane { struct exynos_drm_plane {
struct drm_plane base; struct drm_plane base;
const struct exynos_drm_plane_config *config;
unsigned int zpos; unsigned int zpos;
struct drm_framebuffer *pending_fb; struct drm_framebuffer *pending_fb;
}; };
/*
* Exynos DRM plane configuration structure.
*
* @zpos: z-position of the plane.
* @type: type of the plane (primary, cursor or overlay).
* @pixel_formats: supported pixel formats.
* @num_pixel_formats: number of elements in 'pixel_formats'.
* @capabilities: supported features (see EXYNOS_DRM_PLANE_CAP_*)
*/
struct exynos_drm_plane_config {
unsigned int zpos;
enum drm_plane_type type;
const uint32_t *pixel_formats;
unsigned int num_pixel_formats;
unsigned int capabilities;
};
/* /*
* Exynos drm crtc ops * Exynos drm crtc ops
* *
......
...@@ -88,7 +88,6 @@ ...@@ -88,7 +88,6 @@
/* FIMD has totally five hardware windows. */ /* FIMD has totally five hardware windows. */
#define WINDOWS_NR 5 #define WINDOWS_NR 5
#define CURSOR_WIN 4
struct fimd_driver_data { struct fimd_driver_data {
unsigned int timing_base; unsigned int timing_base;
...@@ -151,6 +150,7 @@ struct fimd_context { ...@@ -151,6 +150,7 @@ struct fimd_context {
struct drm_device *drm_dev; struct drm_device *drm_dev;
struct exynos_drm_crtc *crtc; struct exynos_drm_crtc *crtc;
struct exynos_drm_plane planes[WINDOWS_NR]; struct exynos_drm_plane planes[WINDOWS_NR];
struct exynos_drm_plane_config configs[WINDOWS_NR];
struct clk *bus_clk; struct clk *bus_clk;
struct clk *lcd_clk; struct clk *lcd_clk;
void __iomem *regs; void __iomem *regs;
...@@ -188,6 +188,14 @@ static const struct of_device_id fimd_driver_dt_match[] = { ...@@ -188,6 +188,14 @@ static const struct of_device_id fimd_driver_dt_match[] = {
}; };
MODULE_DEVICE_TABLE(of, fimd_driver_dt_match); MODULE_DEVICE_TABLE(of, fimd_driver_dt_match);
static const enum drm_plane_type fimd_win_types[WINDOWS_NR] = {
DRM_PLANE_TYPE_PRIMARY,
DRM_PLANE_TYPE_OVERLAY,
DRM_PLANE_TYPE_OVERLAY,
DRM_PLANE_TYPE_OVERLAY,
DRM_PLANE_TYPE_CURSOR,
};
static const uint32_t fimd_formats[] = { static const uint32_t fimd_formats[] = {
DRM_FORMAT_C8, DRM_FORMAT_C8,
DRM_FORMAT_XRGB1555, DRM_FORMAT_XRGB1555,
...@@ -927,18 +935,19 @@ static int fimd_bind(struct device *dev, struct device *master, void *data) ...@@ -927,18 +935,19 @@ static int fimd_bind(struct device *dev, struct device *master, void *data)
struct drm_device *drm_dev = data; struct drm_device *drm_dev = data;
struct exynos_drm_private *priv = drm_dev->dev_private; struct exynos_drm_private *priv = drm_dev->dev_private;
struct exynos_drm_plane *exynos_plane; struct exynos_drm_plane *exynos_plane;
enum drm_plane_type type; unsigned int i;
unsigned int zpos;
int ret; int ret;
ctx->drm_dev = drm_dev; ctx->drm_dev = drm_dev;
ctx->pipe = priv->pipe++; ctx->pipe = priv->pipe++;
for (zpos = 0; zpos < WINDOWS_NR; zpos++) { for (i = 0; i < WINDOWS_NR; i++) {
type = exynos_plane_get_type(zpos, CURSOR_WIN); ctx->configs[i].pixel_formats = fimd_formats;
ret = exynos_plane_init(drm_dev, &ctx->planes[zpos], ctx->configs[i].num_pixel_formats = ARRAY_SIZE(fimd_formats);
1 << ctx->pipe, type, fimd_formats, ctx->configs[i].zpos = i;
ARRAY_SIZE(fimd_formats), zpos); ctx->configs[i].type = fimd_win_types[i];
ret = exynos_plane_init(drm_dev, &ctx->planes[i],
1 << ctx->pipe, &ctx->configs[i]);
if (ret) if (ret)
return ret; return ret;
} }
......
...@@ -246,28 +246,20 @@ static void exynos_plane_attach_zpos_property(struct drm_plane *plane, ...@@ -246,28 +246,20 @@ static void exynos_plane_attach_zpos_property(struct drm_plane *plane,
drm_object_attach_property(&plane->base, prop, zpos); drm_object_attach_property(&plane->base, prop, zpos);
} }
enum drm_plane_type exynos_plane_get_type(unsigned int zpos,
unsigned int cursor_win)
{
if (zpos == DEFAULT_WIN)
return DRM_PLANE_TYPE_PRIMARY;
else if (zpos == cursor_win)
return DRM_PLANE_TYPE_CURSOR;
else
return DRM_PLANE_TYPE_OVERLAY;
}
int exynos_plane_init(struct drm_device *dev, int exynos_plane_init(struct drm_device *dev,
struct exynos_drm_plane *exynos_plane, struct exynos_drm_plane *exynos_plane,
unsigned long possible_crtcs, enum drm_plane_type type, unsigned long possible_crtcs,
const uint32_t *formats, unsigned int fcount, const struct exynos_drm_plane_config *config)
unsigned int zpos)
{ {
int err; int err;
err = drm_universal_plane_init(dev, &exynos_plane->base, possible_crtcs, err = drm_universal_plane_init(dev, &exynos_plane->base,
&exynos_plane_funcs, formats, fcount, possible_crtcs,
type); &exynos_plane_funcs,
config->pixel_formats,
config->num_pixel_formats,
config->type);
if (err) { if (err) {
DRM_ERROR("failed to initialize plane\n"); DRM_ERROR("failed to initialize plane\n");
return err; return err;
...@@ -275,10 +267,12 @@ int exynos_plane_init(struct drm_device *dev, ...@@ -275,10 +267,12 @@ int exynos_plane_init(struct drm_device *dev,
drm_plane_helper_add(&exynos_plane->base, &plane_helper_funcs); drm_plane_helper_add(&exynos_plane->base, &plane_helper_funcs);
exynos_plane->zpos = zpos; exynos_plane->zpos = config->zpos;
exynos_plane->config = config;
if (type == DRM_PLANE_TYPE_OVERLAY) if (config->type == DRM_PLANE_TYPE_OVERLAY)
exynos_plane_attach_zpos_property(&exynos_plane->base, zpos); exynos_plane_attach_zpos_property(&exynos_plane->base,
config->zpos);
return 0; return 0;
} }
...@@ -9,10 +9,7 @@ ...@@ -9,10 +9,7 @@
* *
*/ */
enum drm_plane_type exynos_plane_get_type(unsigned int zpos,
unsigned int cursor_win);
int exynos_plane_init(struct drm_device *dev, int exynos_plane_init(struct drm_device *dev,
struct exynos_drm_plane *exynos_plane, struct exynos_drm_plane *exynos_plane,
unsigned long possible_crtcs, enum drm_plane_type type, unsigned long possible_crtcs,
const uint32_t *formats, unsigned int fcount, const struct exynos_drm_plane_config *config);
unsigned int zpos);
...@@ -30,7 +30,6 @@ ...@@ -30,7 +30,6 @@
/* vidi has totally three virtual windows. */ /* vidi has totally three virtual windows. */
#define WINDOWS_NR 3 #define WINDOWS_NR 3
#define CURSOR_WIN 2
#define ctx_from_connector(c) container_of(c, struct vidi_context, \ #define ctx_from_connector(c) container_of(c, struct vidi_context, \
connector) connector)
...@@ -90,6 +89,12 @@ static const uint32_t formats[] = { ...@@ -90,6 +89,12 @@ static const uint32_t formats[] = {
DRM_FORMAT_NV12, DRM_FORMAT_NV12,
}; };
static const enum drm_plane_type vidi_win_types[WINDOWS_NR] = {
DRM_PLANE_TYPE_PRIMARY,
DRM_PLANE_TYPE_OVERLAY,
DRM_PLANE_TYPE_CURSOR,
};
static int vidi_enable_vblank(struct exynos_drm_crtc *crtc) static int vidi_enable_vblank(struct exynos_drm_crtc *crtc)
{ {
struct vidi_context *ctx = crtc->ctx; struct vidi_context *ctx = crtc->ctx;
...@@ -443,17 +448,21 @@ static int vidi_bind(struct device *dev, struct device *master, void *data) ...@@ -443,17 +448,21 @@ static int vidi_bind(struct device *dev, struct device *master, void *data)
struct drm_device *drm_dev = data; struct drm_device *drm_dev = data;
struct drm_encoder *encoder = &ctx->encoder; struct drm_encoder *encoder = &ctx->encoder;
struct exynos_drm_plane *exynos_plane; struct exynos_drm_plane *exynos_plane;
enum drm_plane_type type; struct exynos_drm_plane_config plane_config = { 0 };
unsigned int zpos; unsigned int i;
int pipe, ret; int pipe, ret;
vidi_ctx_initialize(ctx, drm_dev); vidi_ctx_initialize(ctx, drm_dev);
for (zpos = 0; zpos < WINDOWS_NR; zpos++) { plane_config.pixel_formats = formats;
type = exynos_plane_get_type(zpos, CURSOR_WIN); plane_config.num_pixel_formats = ARRAY_SIZE(formats);
ret = exynos_plane_init(drm_dev, &ctx->planes[zpos],
1 << ctx->pipe, type, formats, for (i = 0; i < WINDOWS_NR; i++) {
ARRAY_SIZE(formats), zpos); plane_config.zpos = i;
plane_config.type = vidi_win_types[i];
ret = exynos_plane_init(drm_dev, &ctx->planes[i],
1 << ctx->pipe, &plane_config);
if (ret) if (ret)
return ret; return ret;
} }
......
...@@ -43,7 +43,6 @@ ...@@ -43,7 +43,6 @@
#define MIXER_WIN_NR 3 #define MIXER_WIN_NR 3
#define VP_DEFAULT_WIN 2 #define VP_DEFAULT_WIN 2
#define CURSOR_WIN 1
/* The pixelformats that are natively supported by the mixer. */ /* The pixelformats that are natively supported by the mixer. */
#define MXR_FORMAT_RGB565 4 #define MXR_FORMAT_RGB565 4
...@@ -112,6 +111,25 @@ struct mixer_drv_data { ...@@ -112,6 +111,25 @@ struct mixer_drv_data {
bool has_sclk; bool has_sclk;
}; };
static const struct exynos_drm_plane_config plane_configs[MIXER_WIN_NR] = {
{
.zpos = 0,
.type = DRM_PLANE_TYPE_PRIMARY,
.pixel_formats = mixer_formats,
.num_pixel_formats = ARRAY_SIZE(mixer_formats),
}, {
.zpos = 1,
.type = DRM_PLANE_TYPE_CURSOR,
.pixel_formats = mixer_formats,
.num_pixel_formats = ARRAY_SIZE(mixer_formats),
}, {
.zpos = 2,
.type = DRM_PLANE_TYPE_OVERLAY,
.pixel_formats = vp_formats,
.num_pixel_formats = ARRAY_SIZE(vp_formats),
},
};
static const u8 filter_y_horiz_tap8[] = { static const u8 filter_y_horiz_tap8[] = {
0, -1, -1, -1, -1, -1, -1, -1, 0, -1, -1, -1, -1, -1, -1, -1,
-1, -1, -1, -1, -1, 0, 0, 0, -1, -1, -1, -1, -1, 0, 0, 0,
...@@ -1155,33 +1173,19 @@ static int mixer_bind(struct device *dev, struct device *manager, void *data) ...@@ -1155,33 +1173,19 @@ static int mixer_bind(struct device *dev, struct device *manager, void *data)
struct mixer_context *ctx = dev_get_drvdata(dev); struct mixer_context *ctx = dev_get_drvdata(dev);
struct drm_device *drm_dev = data; struct drm_device *drm_dev = data;
struct exynos_drm_plane *exynos_plane; struct exynos_drm_plane *exynos_plane;
unsigned int zpos; unsigned int i;
int ret; int ret;
ret = mixer_initialize(ctx, drm_dev); ret = mixer_initialize(ctx, drm_dev);
if (ret) if (ret)
return ret; return ret;
for (zpos = 0; zpos < MIXER_WIN_NR; zpos++) { for (i = 0; i < MIXER_WIN_NR; i++) {
enum drm_plane_type type; if (i == VP_DEFAULT_WIN && !ctx->vp_enabled)
const uint32_t *formats;
unsigned int fcount;
if (zpos == VP_DEFAULT_WIN && !ctx->vp_enabled)
continue; continue;
if (zpos < VP_DEFAULT_WIN) { ret = exynos_plane_init(drm_dev, &ctx->planes[i],
formats = mixer_formats; 1 << ctx->pipe, &plane_configs[i]);
fcount = ARRAY_SIZE(mixer_formats);
} else {
formats = vp_formats;
fcount = ARRAY_SIZE(vp_formats);
}
type = exynos_plane_get_type(zpos, CURSOR_WIN);
ret = exynos_plane_init(drm_dev, &ctx->planes[zpos],
1 << ctx->pipe, type, formats, fcount,
zpos);
if (ret) if (ret)
return ret; return ret;
} }
......
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