drm: kirin: Move ade drm init to kirin drm drv
As part of refactoring the kirin driver to better support different hardware revisions, this patch renames ade_data to kirin_drm_private, and moves crtc_init and plane_init to kirin drm drv too. Now that they are generic the functions can be shared between the kirin620 and (to be added later) kirin960 specific support code. Cc: Rongrong Zou <zourongrong@gmail.com> Cc: Xinliang Liu <z.liuxinliang@hisilicon.com> Cc: David Airlie <airlied@linux.ie> Cc: Daniel Vetter <daniel@ffwll.ch> Cc: dri-devel <dri-devel@lists.freedesktop.org> Cc: Sam Ravnborg <sam@ravnborg.org> Acked-by: Xinliang Liu <z.liuxinliang@hisilicon.com> Reviewed-by: Sam Ravnborg <sam@ravnborg.org> Signed-off-by: Xu YiPing <xuyiping@hisilicon.com> [jstultz: Reworded commit message] Signed-off-by: John Stultz <john.stultz@linaro.org> Signed-off-by: Sam Ravnborg <sam@ravnborg.org> Link: https://patchwork.freedesktop.org/patch/msgid/20190820230626.23253-26-john.stultz@linaro.org
This commit is contained in:
parent
28cd05eee4
commit
89a565dba1
|
@ -53,13 +53,6 @@ struct ade_hw_ctx {
|
|||
struct drm_crtc *crtc;
|
||||
};
|
||||
|
||||
struct ade_data {
|
||||
struct kirin_crtc crtc;
|
||||
struct kirin_plane planes[ADE_CH_NUM];
|
||||
struct ade_hw_ctx *hw_ctx;
|
||||
};
|
||||
|
||||
/* ade-format info: */
|
||||
static const struct kirin_format ade_formats[] = {
|
||||
/* 16bpp RGB: */
|
||||
{ DRM_FORMAT_RGB565, ADE_RGB_565 },
|
||||
|
@ -571,36 +564,6 @@ static const struct drm_crtc_funcs ade_crtc_funcs = {
|
|||
.disable_vblank = ade_crtc_disable_vblank,
|
||||
};
|
||||
|
||||
static int kirin_drm_crtc_init(struct drm_device *dev, struct drm_crtc *crtc,
|
||||
struct drm_plane *plane,
|
||||
const struct kirin_drm_data *driver_data)
|
||||
{
|
||||
struct device_node *port;
|
||||
int ret;
|
||||
|
||||
/* set crtc port so that
|
||||
* drm_of_find_possible_crtcs call works
|
||||
*/
|
||||
port = of_get_child_by_name(dev->dev->of_node, "port");
|
||||
if (!port) {
|
||||
DRM_ERROR("no port node found in %pOF\n", dev->dev->of_node);
|
||||
return -EINVAL;
|
||||
}
|
||||
of_node_put(port);
|
||||
crtc->port = port;
|
||||
|
||||
ret = drm_crtc_init_with_planes(dev, crtc, plane, NULL,
|
||||
driver_data->crtc_funcs, NULL);
|
||||
if (ret) {
|
||||
DRM_ERROR("failed to init crtc.\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
drm_crtc_helper_add(crtc, driver_data->crtc_helper_funcs);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void ade_rdma_set(void __iomem *base, struct drm_framebuffer *fb,
|
||||
u32 ch, u32 y, u32 in_h, u32 fmt)
|
||||
{
|
||||
|
@ -893,28 +856,6 @@ static struct drm_plane_funcs ade_plane_funcs = {
|
|||
.atomic_destroy_state = drm_atomic_helper_plane_destroy_state,
|
||||
};
|
||||
|
||||
static int kirin_drm_plane_init(struct drm_device *dev,
|
||||
struct kirin_plane *kplane,
|
||||
enum drm_plane_type type,
|
||||
const struct kirin_drm_data *driver_data)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
ret = drm_universal_plane_init(dev, &kplane->base, 1,
|
||||
driver_data->plane_funcs,
|
||||
driver_data->channel_formats,
|
||||
driver_data->channel_formats_cnt,
|
||||
NULL, type, NULL);
|
||||
if (ret) {
|
||||
DRM_ERROR("fail to init plane, ch=%d\n", kplane->ch);
|
||||
return ret;
|
||||
}
|
||||
|
||||
drm_plane_helper_add(&kplane->base, driver_data->plane_helper_funcs);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void *ade_hw_ctx_alloc(struct platform_device *pdev,
|
||||
struct drm_crtc *crtc)
|
||||
{
|
||||
|
@ -984,73 +925,10 @@ static void *ade_hw_ctx_alloc(struct platform_device *pdev,
|
|||
return ctx;
|
||||
}
|
||||
|
||||
static int ade_drm_init(struct platform_device *pdev)
|
||||
{
|
||||
struct drm_device *dev = platform_get_drvdata(pdev);
|
||||
struct ade_data *ade;
|
||||
struct ade_hw_ctx *ctx;
|
||||
struct kirin_crtc *kcrtc;
|
||||
struct kirin_plane *kplane;
|
||||
enum drm_plane_type type;
|
||||
int prim_plane;
|
||||
int ret;
|
||||
u32 ch;
|
||||
|
||||
ade = devm_kzalloc(dev->dev, sizeof(*ade), GFP_KERNEL);
|
||||
if (!ade) {
|
||||
DRM_ERROR("failed to alloc ade_data\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
ctx = ade_driver_data.alloc_hw_ctx(pdev, &ade->crtc.base);
|
||||
if (IS_ERR(ctx)) {
|
||||
DRM_ERROR("failed to initialize kirin_priv hw ctx\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
ade->hw_ctx = ctx;
|
||||
|
||||
kcrtc = &ade->crtc;
|
||||
kcrtc->hw_ctx = ctx;
|
||||
|
||||
/*
|
||||
* plane init
|
||||
* TODO: Now only support primary plane, overlay planes
|
||||
* need to do.
|
||||
*/
|
||||
for (ch = 0; ch < ade_driver_data.num_planes; ch++) {
|
||||
kplane = &ade->planes[ch];
|
||||
kplane->ch = ch;
|
||||
kplane->hw_ctx = ctx;
|
||||
|
||||
if (ch == ade_driver_data.prim_plane)
|
||||
type = DRM_PLANE_TYPE_PRIMARY;
|
||||
else
|
||||
type = DRM_PLANE_TYPE_OVERLAY;
|
||||
|
||||
ret = kirin_drm_plane_init(dev, kplane, type, &ade_driver_data);
|
||||
if (ret)
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* crtc init */
|
||||
prim_plane = ade_driver_data.prim_plane;
|
||||
ret = kirin_drm_crtc_init(dev, &kcrtc->base,
|
||||
&ade->planes[prim_plane].base,
|
||||
&ade_driver_data);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void ade_hw_ctx_cleanup(void *hw_ctx)
|
||||
{
|
||||
}
|
||||
|
||||
static void ade_drm_cleanup(struct platform_device *pdev)
|
||||
{
|
||||
}
|
||||
|
||||
static const struct drm_mode_config_funcs ade_mode_config_funcs = {
|
||||
.fb_create = drm_gem_fb_create,
|
||||
.atomic_check = drm_atomic_helper_check,
|
||||
|
@ -1098,7 +976,4 @@ struct kirin_drm_data ade_driver_data = {
|
|||
|
||||
.alloc_hw_ctx = ade_hw_ctx_alloc,
|
||||
.cleanup_hw_ctx = ade_hw_ctx_cleanup,
|
||||
|
||||
.init = ade_drm_init,
|
||||
.cleanup = ade_drm_cleanup
|
||||
};
|
||||
|
|
|
@ -29,6 +29,130 @@
|
|||
|
||||
#include "kirin_drm_drv.h"
|
||||
|
||||
#define KIRIN_MAX_PLANE 2
|
||||
|
||||
struct kirin_drm_private {
|
||||
struct kirin_crtc crtc;
|
||||
struct kirin_plane planes[KIRIN_MAX_PLANE];
|
||||
void *hw_ctx;
|
||||
};
|
||||
|
||||
static int kirin_drm_crtc_init(struct drm_device *dev, struct drm_crtc *crtc,
|
||||
struct drm_plane *plane,
|
||||
const struct kirin_drm_data *driver_data)
|
||||
{
|
||||
struct device_node *port;
|
||||
int ret;
|
||||
|
||||
/* set crtc port so that
|
||||
* drm_of_find_possible_crtcs call works
|
||||
*/
|
||||
port = of_get_child_by_name(dev->dev->of_node, "port");
|
||||
if (!port) {
|
||||
DRM_ERROR("no port node found in %pOF\n", dev->dev->of_node);
|
||||
return -EINVAL;
|
||||
}
|
||||
of_node_put(port);
|
||||
crtc->port = port;
|
||||
|
||||
ret = drm_crtc_init_with_planes(dev, crtc, plane, NULL,
|
||||
driver_data->crtc_funcs, NULL);
|
||||
if (ret) {
|
||||
DRM_ERROR("failed to init crtc.\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
drm_crtc_helper_add(crtc, driver_data->crtc_helper_funcs);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int kirin_drm_plane_init(struct drm_device *dev, struct drm_plane *plane,
|
||||
enum drm_plane_type type,
|
||||
const struct kirin_drm_data *data)
|
||||
{
|
||||
int ret = 0;
|
||||
|
||||
ret = drm_universal_plane_init(dev, plane, 1, data->plane_funcs,
|
||||
data->channel_formats,
|
||||
data->channel_formats_cnt,
|
||||
NULL, type, NULL);
|
||||
if (ret) {
|
||||
DRM_ERROR("fail to init plane, ch=%d\n", 0);
|
||||
return ret;
|
||||
}
|
||||
|
||||
drm_plane_helper_add(plane, data->plane_helper_funcs);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void kirin_drm_private_cleanup(struct drm_device *dev)
|
||||
{
|
||||
struct kirin_drm_private *kirin_priv = dev->dev_private;
|
||||
struct kirin_drm_data *data;
|
||||
|
||||
data = (struct kirin_drm_data *)of_device_get_match_data(dev->dev);
|
||||
if (data->cleanup_hw_ctx)
|
||||
data->cleanup_hw_ctx(kirin_priv->hw_ctx);
|
||||
|
||||
devm_kfree(dev->dev, kirin_priv);
|
||||
dev->dev_private = NULL;
|
||||
}
|
||||
|
||||
static int kirin_drm_private_init(struct drm_device *dev,
|
||||
const struct kirin_drm_data *driver_data)
|
||||
{
|
||||
struct platform_device *pdev = to_platform_device(dev->dev);
|
||||
struct kirin_drm_private *kirin_priv;
|
||||
struct drm_plane *prim_plane;
|
||||
enum drm_plane_type type;
|
||||
void *ctx;
|
||||
int ret;
|
||||
u32 ch;
|
||||
|
||||
kirin_priv = devm_kzalloc(dev->dev, sizeof(*kirin_priv), GFP_KERNEL);
|
||||
if (!kirin_priv) {
|
||||
DRM_ERROR("failed to alloc kirin_drm_private\n");
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
ctx = driver_data->alloc_hw_ctx(pdev, &kirin_priv->crtc.base);
|
||||
if (IS_ERR(ctx)) {
|
||||
DRM_ERROR("failed to initialize kirin_priv hw ctx\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
kirin_priv->hw_ctx = ctx;
|
||||
|
||||
/*
|
||||
* plane init
|
||||
* TODO: Now only support primary plane, overlay planes
|
||||
* need to do.
|
||||
*/
|
||||
for (ch = 0; ch < driver_data->num_planes; ch++) {
|
||||
if (ch == driver_data->prim_plane)
|
||||
type = DRM_PLANE_TYPE_PRIMARY;
|
||||
else
|
||||
type = DRM_PLANE_TYPE_OVERLAY;
|
||||
ret = kirin_drm_plane_init(dev, &kirin_priv->planes[ch].base,
|
||||
type, driver_data);
|
||||
if (ret)
|
||||
return ret;
|
||||
kirin_priv->planes[ch].ch = ch;
|
||||
kirin_priv->planes[ch].hw_ctx = ctx;
|
||||
}
|
||||
|
||||
/* crtc init */
|
||||
prim_plane = &kirin_priv->planes[driver_data->prim_plane].base;
|
||||
ret = kirin_drm_crtc_init(dev, &kirin_priv->crtc.base,
|
||||
prim_plane, driver_data);
|
||||
if (ret)
|
||||
return ret;
|
||||
kirin_priv->crtc.hw_ctx = ctx;
|
||||
dev->dev_private = kirin_priv;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int kirin_drm_kms_init(struct drm_device *dev,
|
||||
const struct kirin_drm_data *driver_data)
|
||||
|
@ -44,7 +168,7 @@ static int kirin_drm_kms_init(struct drm_device *dev,
|
|||
dev->mode_config.funcs = driver_data->mode_config_funcs;
|
||||
|
||||
/* display controller init */
|
||||
ret = driver_data->init(to_platform_device(dev->dev));
|
||||
ret = kirin_drm_private_init(dev, driver_data);
|
||||
if (ret)
|
||||
goto err_mode_config_cleanup;
|
||||
|
||||
|
@ -52,7 +176,7 @@ static int kirin_drm_kms_init(struct drm_device *dev,
|
|||
ret = component_bind_all(dev->dev, dev);
|
||||
if (ret) {
|
||||
DRM_ERROR("failed to bind all component.\n");
|
||||
goto err_dc_cleanup;
|
||||
goto err_private_cleanup;
|
||||
}
|
||||
|
||||
/* vblank init */
|
||||
|
@ -74,11 +198,10 @@ static int kirin_drm_kms_init(struct drm_device *dev,
|
|||
|
||||
err_unbind_all:
|
||||
component_unbind_all(dev->dev, dev);
|
||||
err_dc_cleanup:
|
||||
driver_data->cleanup(to_platform_device(dev->dev));
|
||||
err_private_cleanup:
|
||||
kirin_drm_private_cleanup(dev);
|
||||
err_mode_config_cleanup:
|
||||
drm_mode_config_cleanup(dev);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
|
@ -89,14 +212,8 @@ static int compare_of(struct device *dev, void *data)
|
|||
|
||||
static int kirin_drm_kms_cleanup(struct drm_device *dev)
|
||||
{
|
||||
const struct kirin_drm_data *driver_data;
|
||||
|
||||
drm_kms_helper_poll_fini(dev);
|
||||
|
||||
driver_data = of_device_get_match_data(dev->dev);
|
||||
if (driver_data->cleanup)
|
||||
driver_data->cleanup(to_platform_device(dev->dev));
|
||||
|
||||
kirin_drm_private_cleanup(dev);
|
||||
drm_mode_config_cleanup(dev);
|
||||
|
||||
return 0;
|
||||
|
|
|
@ -7,8 +7,6 @@
|
|||
#ifndef __KIRIN_DRM_DRV_H__
|
||||
#define __KIRIN_DRM_DRV_H__
|
||||
|
||||
#define MAX_CRTC 2
|
||||
|
||||
#define to_kirin_crtc(crtc) \
|
||||
container_of(crtc, struct kirin_crtc, base)
|
||||
|
||||
|
@ -53,9 +51,6 @@ struct kirin_drm_data {
|
|||
void *(*alloc_hw_ctx)(struct platform_device *pdev,
|
||||
struct drm_crtc *crtc);
|
||||
void (*cleanup_hw_ctx)(void *hw_ctx);
|
||||
|
||||
int (*init)(struct platform_device *pdev);
|
||||
void (*cleanup)(struct platform_device *pdev);
|
||||
};
|
||||
|
||||
extern struct kirin_drm_data ade_driver_data;
|
||||
|
|
Loading…
Reference in New Issue