6e6cf3e2f2
Currently both rockchip_drm_psr_activate() and _deactivate() only set the boolean "active" flag without actually making sure that hardware state complies with it. Since we are going to extend the usage of this API to properly lock PSR for the duration of atomic commits, we change the semantics in following way: - a counter is used to track the number of inhibit requests, - PSR is actually disabled in hardware on first inhibit request, - PSR enable work is scheduled on last allow request. The above allows using the API as a way to deterministically synchronize PSR state changes with other DRM events, i.e. atomic commits and cursor updates. As a nice side effect, the naming is sorted out and we have "inhibit" for stopping the software logic and "enable" for hardware state. Signed-off-by: Tomasz Figa <tfiga@chromium.org> Signed-off-by: Thierry Escande <thierry.escande@collabora.com> Signed-off-by: Enric Balletbo i Serra <enric.balletbo@collabora.com> Tested-by: Marek Szyprowski <m.szyprowski@samsung.com> Reviewed-by: Andrzej Hajda <a.hajda@samsung.com> Signed-off-by: Andrzej Hajda <a.hajda@samsung.com> Link: https://patchwork.freedesktop.org/patch/msgid/20180423105003.9004-26-enric.balletbo@collabora.com
474 lines
12 KiB
C
474 lines
12 KiB
C
/*
|
|
* Rockchip SoC DP (Display Port) interface driver.
|
|
*
|
|
* Copyright (C) Fuzhou Rockchip Electronics Co., Ltd.
|
|
* Author: Andy Yan <andy.yan@rock-chips.com>
|
|
* Yakir Yang <ykk@rock-chips.com>
|
|
* Jeff Chen <jeff.chen@rock-chips.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 <linux/component.h>
|
|
#include <linux/mfd/syscon.h>
|
|
#include <linux/of_device.h>
|
|
#include <linux/of_graph.h>
|
|
#include <linux/regmap.h>
|
|
#include <linux/reset.h>
|
|
#include <linux/clk.h>
|
|
|
|
#include <drm/drmP.h>
|
|
#include <drm/drm_crtc_helper.h>
|
|
#include <drm/drm_dp_helper.h>
|
|
#include <drm/drm_of.h>
|
|
#include <drm/drm_panel.h>
|
|
|
|
#include <video/of_videomode.h>
|
|
#include <video/videomode.h>
|
|
|
|
#include <drm/bridge/analogix_dp.h>
|
|
|
|
#include "rockchip_drm_drv.h"
|
|
#include "rockchip_drm_psr.h"
|
|
#include "rockchip_drm_vop.h"
|
|
|
|
#define RK3288_GRF_SOC_CON6 0x25c
|
|
#define RK3288_EDP_LCDC_SEL BIT(5)
|
|
#define RK3399_GRF_SOC_CON20 0x6250
|
|
#define RK3399_EDP_LCDC_SEL BIT(5)
|
|
|
|
#define HIWORD_UPDATE(val, mask) (val | (mask) << 16)
|
|
|
|
#define PSR_WAIT_LINE_FLAG_TIMEOUT_MS 100
|
|
|
|
#define to_dp(nm) container_of(nm, struct rockchip_dp_device, nm)
|
|
|
|
/**
|
|
* struct rockchip_dp_chip_data - splite the grf setting of kind of chips
|
|
* @lcdsel_grf_reg: grf register offset of lcdc select
|
|
* @lcdsel_big: reg value of selecting vop big for eDP
|
|
* @lcdsel_lit: reg value of selecting vop little for eDP
|
|
* @chip_type: specific chip type
|
|
*/
|
|
struct rockchip_dp_chip_data {
|
|
u32 lcdsel_grf_reg;
|
|
u32 lcdsel_big;
|
|
u32 lcdsel_lit;
|
|
u32 chip_type;
|
|
};
|
|
|
|
struct rockchip_dp_device {
|
|
struct drm_device *drm_dev;
|
|
struct device *dev;
|
|
struct drm_encoder encoder;
|
|
struct drm_display_mode mode;
|
|
|
|
struct clk *pclk;
|
|
struct clk *grfclk;
|
|
struct regmap *grf;
|
|
struct reset_control *rst;
|
|
|
|
const struct rockchip_dp_chip_data *data;
|
|
|
|
struct analogix_dp_device *adp;
|
|
struct analogix_dp_plat_data plat_data;
|
|
};
|
|
|
|
static int analogix_dp_psr_set(struct drm_encoder *encoder, bool enabled)
|
|
{
|
|
struct rockchip_dp_device *dp = to_dp(encoder);
|
|
int ret;
|
|
|
|
if (!analogix_dp_psr_enabled(dp->adp))
|
|
return 0;
|
|
|
|
DRM_DEV_DEBUG(dp->dev, "%s PSR...\n", enabled ? "Entry" : "Exit");
|
|
|
|
ret = rockchip_drm_wait_vact_end(dp->encoder.crtc,
|
|
PSR_WAIT_LINE_FLAG_TIMEOUT_MS);
|
|
if (ret) {
|
|
DRM_DEV_ERROR(dp->dev, "line flag interrupt did not arrive\n");
|
|
return -ETIMEDOUT;
|
|
}
|
|
|
|
if (enabled)
|
|
return analogix_dp_enable_psr(dp->adp);
|
|
else
|
|
return analogix_dp_disable_psr(dp->adp);
|
|
}
|
|
|
|
static int rockchip_dp_pre_init(struct rockchip_dp_device *dp)
|
|
{
|
|
reset_control_assert(dp->rst);
|
|
usleep_range(10, 20);
|
|
reset_control_deassert(dp->rst);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int rockchip_dp_poweron_start(struct analogix_dp_plat_data *plat_data)
|
|
{
|
|
struct rockchip_dp_device *dp = to_dp(plat_data);
|
|
int ret;
|
|
|
|
ret = clk_prepare_enable(dp->pclk);
|
|
if (ret < 0) {
|
|
DRM_DEV_ERROR(dp->dev, "failed to enable pclk %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
ret = rockchip_dp_pre_init(dp);
|
|
if (ret < 0) {
|
|
DRM_DEV_ERROR(dp->dev, "failed to dp pre init %d\n", ret);
|
|
clk_disable_unprepare(dp->pclk);
|
|
return ret;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int rockchip_dp_poweron_end(struct analogix_dp_plat_data *plat_data)
|
|
{
|
|
struct rockchip_dp_device *dp = to_dp(plat_data);
|
|
|
|
return rockchip_drm_psr_inhibit_put(&dp->encoder);
|
|
}
|
|
|
|
static int rockchip_dp_powerdown(struct analogix_dp_plat_data *plat_data)
|
|
{
|
|
struct rockchip_dp_device *dp = to_dp(plat_data);
|
|
int ret;
|
|
|
|
ret = rockchip_drm_psr_inhibit_get(&dp->encoder);
|
|
if (ret != 0)
|
|
return ret;
|
|
|
|
clk_disable_unprepare(dp->pclk);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int rockchip_dp_get_modes(struct analogix_dp_plat_data *plat_data,
|
|
struct drm_connector *connector)
|
|
{
|
|
struct drm_display_info *di = &connector->display_info;
|
|
/* VOP couldn't output YUV video format for eDP rightly */
|
|
u32 mask = DRM_COLOR_FORMAT_YCRCB444 | DRM_COLOR_FORMAT_YCRCB422;
|
|
|
|
if ((di->color_formats & mask)) {
|
|
DRM_DEBUG_KMS("Swapping display color format from YUV to RGB\n");
|
|
di->color_formats &= ~mask;
|
|
di->color_formats |= DRM_COLOR_FORMAT_RGB444;
|
|
di->bpc = 8;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static bool
|
|
rockchip_dp_drm_encoder_mode_fixup(struct drm_encoder *encoder,
|
|
const struct drm_display_mode *mode,
|
|
struct drm_display_mode *adjusted_mode)
|
|
{
|
|
/* do nothing */
|
|
return true;
|
|
}
|
|
|
|
static void rockchip_dp_drm_encoder_mode_set(struct drm_encoder *encoder,
|
|
struct drm_display_mode *mode,
|
|
struct drm_display_mode *adjusted)
|
|
{
|
|
/* do nothing */
|
|
}
|
|
|
|
static void rockchip_dp_drm_encoder_enable(struct drm_encoder *encoder)
|
|
{
|
|
struct rockchip_dp_device *dp = to_dp(encoder);
|
|
int ret;
|
|
u32 val;
|
|
|
|
ret = drm_of_encoder_active_endpoint_id(dp->dev->of_node, encoder);
|
|
if (ret < 0)
|
|
return;
|
|
|
|
if (ret)
|
|
val = dp->data->lcdsel_lit;
|
|
else
|
|
val = dp->data->lcdsel_big;
|
|
|
|
DRM_DEV_DEBUG(dp->dev, "vop %s output to dp\n", (ret) ? "LIT" : "BIG");
|
|
|
|
ret = clk_prepare_enable(dp->grfclk);
|
|
if (ret < 0) {
|
|
DRM_DEV_ERROR(dp->dev, "failed to enable grfclk %d\n", ret);
|
|
return;
|
|
}
|
|
|
|
ret = regmap_write(dp->grf, dp->data->lcdsel_grf_reg, val);
|
|
if (ret != 0)
|
|
DRM_DEV_ERROR(dp->dev, "Could not write to GRF: %d\n", ret);
|
|
|
|
clk_disable_unprepare(dp->grfclk);
|
|
}
|
|
|
|
static void rockchip_dp_drm_encoder_nop(struct drm_encoder *encoder)
|
|
{
|
|
/* do nothing */
|
|
}
|
|
|
|
static int
|
|
rockchip_dp_drm_encoder_atomic_check(struct drm_encoder *encoder,
|
|
struct drm_crtc_state *crtc_state,
|
|
struct drm_connector_state *conn_state)
|
|
{
|
|
struct rockchip_crtc_state *s = to_rockchip_crtc_state(crtc_state);
|
|
struct drm_display_info *di = &conn_state->connector->display_info;
|
|
|
|
/*
|
|
* The hardware IC designed that VOP must output the RGB10 video
|
|
* format to eDP controller, and if eDP panel only support RGB8,
|
|
* then eDP controller should cut down the video data, not via VOP
|
|
* controller, that's why we need to hardcode the VOP output mode
|
|
* to RGA10 here.
|
|
*/
|
|
|
|
s->output_mode = ROCKCHIP_OUT_MODE_AAAA;
|
|
s->output_type = DRM_MODE_CONNECTOR_eDP;
|
|
s->output_bpc = di->bpc;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct drm_encoder_helper_funcs rockchip_dp_encoder_helper_funcs = {
|
|
.mode_fixup = rockchip_dp_drm_encoder_mode_fixup,
|
|
.mode_set = rockchip_dp_drm_encoder_mode_set,
|
|
.enable = rockchip_dp_drm_encoder_enable,
|
|
.disable = rockchip_dp_drm_encoder_nop,
|
|
.atomic_check = rockchip_dp_drm_encoder_atomic_check,
|
|
};
|
|
|
|
static struct drm_encoder_funcs rockchip_dp_encoder_funcs = {
|
|
.destroy = drm_encoder_cleanup,
|
|
};
|
|
|
|
static int rockchip_dp_of_probe(struct rockchip_dp_device *dp)
|
|
{
|
|
struct device *dev = dp->dev;
|
|
struct device_node *np = dev->of_node;
|
|
|
|
dp->grf = syscon_regmap_lookup_by_phandle(np, "rockchip,grf");
|
|
if (IS_ERR(dp->grf)) {
|
|
DRM_DEV_ERROR(dev, "failed to get rockchip,grf property\n");
|
|
return PTR_ERR(dp->grf);
|
|
}
|
|
|
|
dp->grfclk = devm_clk_get(dev, "grf");
|
|
if (PTR_ERR(dp->grfclk) == -ENOENT) {
|
|
dp->grfclk = NULL;
|
|
} else if (PTR_ERR(dp->grfclk) == -EPROBE_DEFER) {
|
|
return -EPROBE_DEFER;
|
|
} else if (IS_ERR(dp->grfclk)) {
|
|
DRM_DEV_ERROR(dev, "failed to get grf clock\n");
|
|
return PTR_ERR(dp->grfclk);
|
|
}
|
|
|
|
dp->pclk = devm_clk_get(dev, "pclk");
|
|
if (IS_ERR(dp->pclk)) {
|
|
DRM_DEV_ERROR(dev, "failed to get pclk property\n");
|
|
return PTR_ERR(dp->pclk);
|
|
}
|
|
|
|
dp->rst = devm_reset_control_get(dev, "dp");
|
|
if (IS_ERR(dp->rst)) {
|
|
DRM_DEV_ERROR(dev, "failed to get dp reset control\n");
|
|
return PTR_ERR(dp->rst);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int rockchip_dp_drm_create_encoder(struct rockchip_dp_device *dp)
|
|
{
|
|
struct drm_encoder *encoder = &dp->encoder;
|
|
struct drm_device *drm_dev = dp->drm_dev;
|
|
struct device *dev = dp->dev;
|
|
int ret;
|
|
|
|
encoder->possible_crtcs = drm_of_find_possible_crtcs(drm_dev,
|
|
dev->of_node);
|
|
DRM_DEBUG_KMS("possible_crtcs = 0x%x\n", encoder->possible_crtcs);
|
|
|
|
ret = drm_encoder_init(drm_dev, encoder, &rockchip_dp_encoder_funcs,
|
|
DRM_MODE_ENCODER_TMDS, NULL);
|
|
if (ret) {
|
|
DRM_ERROR("failed to initialize encoder with drm\n");
|
|
return ret;
|
|
}
|
|
|
|
drm_encoder_helper_add(encoder, &rockchip_dp_encoder_helper_funcs);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int rockchip_dp_bind(struct device *dev, struct device *master,
|
|
void *data)
|
|
{
|
|
struct rockchip_dp_device *dp = dev_get_drvdata(dev);
|
|
const struct rockchip_dp_chip_data *dp_data;
|
|
struct drm_device *drm_dev = data;
|
|
int ret;
|
|
|
|
dp_data = of_device_get_match_data(dev);
|
|
if (!dp_data)
|
|
return -ENODEV;
|
|
|
|
dp->data = dp_data;
|
|
dp->drm_dev = drm_dev;
|
|
|
|
ret = rockchip_dp_drm_create_encoder(dp);
|
|
if (ret) {
|
|
DRM_ERROR("failed to create drm encoder\n");
|
|
return ret;
|
|
}
|
|
|
|
dp->plat_data.encoder = &dp->encoder;
|
|
|
|
dp->plat_data.dev_type = dp->data->chip_type;
|
|
dp->plat_data.power_on_start = rockchip_dp_poweron_start;
|
|
dp->plat_data.power_on_end = rockchip_dp_poweron_end;
|
|
dp->plat_data.power_off = rockchip_dp_powerdown;
|
|
dp->plat_data.get_modes = rockchip_dp_get_modes;
|
|
|
|
ret = rockchip_drm_psr_register(&dp->encoder, analogix_dp_psr_set);
|
|
if (ret < 0)
|
|
goto err_cleanup_encoder;
|
|
|
|
dp->adp = analogix_dp_bind(dev, dp->drm_dev, &dp->plat_data);
|
|
if (IS_ERR(dp->adp)) {
|
|
ret = PTR_ERR(dp->adp);
|
|
goto err_unreg_psr;
|
|
}
|
|
|
|
return 0;
|
|
err_unreg_psr:
|
|
rockchip_drm_psr_unregister(&dp->encoder);
|
|
err_cleanup_encoder:
|
|
dp->encoder.funcs->destroy(&dp->encoder);
|
|
return ret;
|
|
}
|
|
|
|
static void rockchip_dp_unbind(struct device *dev, struct device *master,
|
|
void *data)
|
|
{
|
|
struct rockchip_dp_device *dp = dev_get_drvdata(dev);
|
|
|
|
analogix_dp_unbind(dp->adp);
|
|
rockchip_drm_psr_unregister(&dp->encoder);
|
|
dp->encoder.funcs->destroy(&dp->encoder);
|
|
|
|
dp->adp = ERR_PTR(-ENODEV);
|
|
}
|
|
|
|
static const struct component_ops rockchip_dp_component_ops = {
|
|
.bind = rockchip_dp_bind,
|
|
.unbind = rockchip_dp_unbind,
|
|
};
|
|
|
|
static int rockchip_dp_probe(struct platform_device *pdev)
|
|
{
|
|
struct device *dev = &pdev->dev;
|
|
struct drm_panel *panel = NULL;
|
|
struct rockchip_dp_device *dp;
|
|
int ret;
|
|
|
|
ret = drm_of_find_panel_or_bridge(dev->of_node, 1, 0, &panel, NULL);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
dp = devm_kzalloc(dev, sizeof(*dp), GFP_KERNEL);
|
|
if (!dp)
|
|
return -ENOMEM;
|
|
|
|
dp->dev = dev;
|
|
dp->adp = ERR_PTR(-ENODEV);
|
|
dp->plat_data.panel = panel;
|
|
|
|
ret = rockchip_dp_of_probe(dp);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
platform_set_drvdata(pdev, dp);
|
|
|
|
return component_add(dev, &rockchip_dp_component_ops);
|
|
}
|
|
|
|
static int rockchip_dp_remove(struct platform_device *pdev)
|
|
{
|
|
component_del(&pdev->dev, &rockchip_dp_component_ops);
|
|
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CONFIG_PM_SLEEP
|
|
static int rockchip_dp_suspend(struct device *dev)
|
|
{
|
|
struct rockchip_dp_device *dp = dev_get_drvdata(dev);
|
|
|
|
if (IS_ERR(dp->adp))
|
|
return 0;
|
|
|
|
return analogix_dp_suspend(dp->adp);
|
|
}
|
|
|
|
static int rockchip_dp_resume(struct device *dev)
|
|
{
|
|
struct rockchip_dp_device *dp = dev_get_drvdata(dev);
|
|
|
|
if (IS_ERR(dp->adp))
|
|
return 0;
|
|
|
|
return analogix_dp_resume(dp->adp);
|
|
}
|
|
#endif
|
|
|
|
static const struct dev_pm_ops rockchip_dp_pm_ops = {
|
|
#ifdef CONFIG_PM_SLEEP
|
|
.suspend = rockchip_dp_suspend,
|
|
.resume_early = rockchip_dp_resume,
|
|
#endif
|
|
};
|
|
|
|
static const struct rockchip_dp_chip_data rk3399_edp = {
|
|
.lcdsel_grf_reg = RK3399_GRF_SOC_CON20,
|
|
.lcdsel_big = HIWORD_UPDATE(0, RK3399_EDP_LCDC_SEL),
|
|
.lcdsel_lit = HIWORD_UPDATE(RK3399_EDP_LCDC_SEL, RK3399_EDP_LCDC_SEL),
|
|
.chip_type = RK3399_EDP,
|
|
};
|
|
|
|
static const struct rockchip_dp_chip_data rk3288_dp = {
|
|
.lcdsel_grf_reg = RK3288_GRF_SOC_CON6,
|
|
.lcdsel_big = HIWORD_UPDATE(0, RK3288_EDP_LCDC_SEL),
|
|
.lcdsel_lit = HIWORD_UPDATE(RK3288_EDP_LCDC_SEL, RK3288_EDP_LCDC_SEL),
|
|
.chip_type = RK3288_DP,
|
|
};
|
|
|
|
static const struct of_device_id rockchip_dp_dt_ids[] = {
|
|
{.compatible = "rockchip,rk3288-dp", .data = &rk3288_dp },
|
|
{.compatible = "rockchip,rk3399-edp", .data = &rk3399_edp },
|
|
{}
|
|
};
|
|
MODULE_DEVICE_TABLE(of, rockchip_dp_dt_ids);
|
|
|
|
struct platform_driver rockchip_dp_driver = {
|
|
.probe = rockchip_dp_probe,
|
|
.remove = rockchip_dp_remove,
|
|
.driver = {
|
|
.name = "rockchip-dp",
|
|
.pm = &rockchip_dp_pm_ops,
|
|
.of_match_table = of_match_ptr(rockchip_dp_dt_ids),
|
|
},
|
|
};
|