You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
652 lines
14 KiB
652 lines
14 KiB
/*
|
|
* Copyright (c) 2017-2019, The Linux Foundation. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 and
|
|
* only version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
*/
|
|
|
|
#define pr_fmt(fmt) "[drm-dp]: %s: " fmt, __func__
|
|
|
|
#include <drm/drm_atomic_helper.h>
|
|
#include <drm/drm_atomic.h>
|
|
#include <drm/drm_crtc.h>
|
|
|
|
#include "msm_drv.h"
|
|
#include "msm_kms.h"
|
|
#include "sde_connector.h"
|
|
#include "dp_drm.h"
|
|
#include "dp_debug.h"
|
|
|
|
#define DP_MST_DEBUG(fmt, ...) pr_debug(fmt, ##__VA_ARGS__)
|
|
|
|
#define to_dp_bridge(x) container_of((x), struct dp_bridge, base)
|
|
|
|
void convert_to_drm_mode(const struct dp_display_mode *dp_mode,
|
|
struct drm_display_mode *drm_mode)
|
|
{
|
|
u32 flags = 0;
|
|
|
|
memset(drm_mode, 0, sizeof(*drm_mode));
|
|
|
|
drm_mode->hdisplay = dp_mode->timing.h_active;
|
|
drm_mode->hsync_start = drm_mode->hdisplay +
|
|
dp_mode->timing.h_front_porch;
|
|
drm_mode->hsync_end = drm_mode->hsync_start +
|
|
dp_mode->timing.h_sync_width;
|
|
drm_mode->htotal = drm_mode->hsync_end + dp_mode->timing.h_back_porch;
|
|
drm_mode->hskew = dp_mode->timing.h_skew;
|
|
|
|
drm_mode->vdisplay = dp_mode->timing.v_active;
|
|
drm_mode->vsync_start = drm_mode->vdisplay +
|
|
dp_mode->timing.v_front_porch;
|
|
drm_mode->vsync_end = drm_mode->vsync_start +
|
|
dp_mode->timing.v_sync_width;
|
|
drm_mode->vtotal = drm_mode->vsync_end + dp_mode->timing.v_back_porch;
|
|
|
|
drm_mode->vrefresh = dp_mode->timing.refresh_rate;
|
|
drm_mode->clock = dp_mode->timing.pixel_clk_khz;
|
|
|
|
if (dp_mode->timing.h_active_low)
|
|
flags |= DRM_MODE_FLAG_NHSYNC;
|
|
else
|
|
flags |= DRM_MODE_FLAG_PHSYNC;
|
|
|
|
if (dp_mode->timing.v_active_low)
|
|
flags |= DRM_MODE_FLAG_NVSYNC;
|
|
else
|
|
flags |= DRM_MODE_FLAG_PVSYNC;
|
|
|
|
drm_mode->flags = flags;
|
|
|
|
drm_mode->type = 0x48;
|
|
drm_mode_set_name(drm_mode);
|
|
}
|
|
|
|
static int dp_bridge_attach(struct drm_bridge *dp_bridge)
|
|
{
|
|
struct dp_bridge *bridge = to_dp_bridge(dp_bridge);
|
|
|
|
if (!dp_bridge) {
|
|
pr_err("Invalid params\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
pr_debug("[%d] attached\n", bridge->id);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void dp_bridge_pre_enable(struct drm_bridge *drm_bridge)
|
|
{
|
|
int rc = 0;
|
|
struct dp_bridge *bridge;
|
|
struct dp_display *dp;
|
|
|
|
if (!drm_bridge) {
|
|
pr_err("Invalid params\n");
|
|
return;
|
|
}
|
|
|
|
bridge = to_dp_bridge(drm_bridge);
|
|
dp = bridge->display;
|
|
|
|
if (!bridge->connector) {
|
|
pr_err("Invalid connector\n");
|
|
return;
|
|
}
|
|
|
|
if (!bridge->dp_panel) {
|
|
pr_err("Invalid dp_panel\n");
|
|
return;
|
|
}
|
|
|
|
/* By this point mode should have been validated through mode_fixup */
|
|
rc = dp->set_mode(dp, bridge->dp_panel, &bridge->dp_mode);
|
|
if (rc) {
|
|
pr_err("[%d] failed to perform a mode set, rc=%d\n",
|
|
bridge->id, rc);
|
|
return;
|
|
}
|
|
|
|
rc = dp->prepare(dp, bridge->dp_panel);
|
|
if (rc) {
|
|
pr_err("[%d] DP display prepare failed, rc=%d\n",
|
|
bridge->id, rc);
|
|
return;
|
|
}
|
|
|
|
/* for SST force stream id, start slot and total slots to 0 */
|
|
dp->set_stream_info(dp, bridge->dp_panel, 0, 0, 0, 0, 0);
|
|
|
|
rc = dp->enable(dp, bridge->dp_panel);
|
|
if (rc) {
|
|
pr_err("[%d] DP display enable failed, rc=%d\n",
|
|
bridge->id, rc);
|
|
dp->unprepare(dp, bridge->dp_panel);
|
|
}
|
|
}
|
|
|
|
static void dp_bridge_enable(struct drm_bridge *drm_bridge)
|
|
{
|
|
int rc = 0;
|
|
struct dp_bridge *bridge;
|
|
struct dp_display *dp;
|
|
|
|
if (!drm_bridge) {
|
|
pr_err("Invalid params\n");
|
|
return;
|
|
}
|
|
|
|
bridge = to_dp_bridge(drm_bridge);
|
|
if (!bridge->connector) {
|
|
pr_err("Invalid connector\n");
|
|
return;
|
|
}
|
|
|
|
if (!bridge->dp_panel) {
|
|
pr_err("Invalid dp_panel\n");
|
|
return;
|
|
}
|
|
|
|
dp = bridge->display;
|
|
|
|
rc = dp->post_enable(dp, bridge->dp_panel);
|
|
if (rc)
|
|
pr_err("[%d] DP display post enable failed, rc=%d\n",
|
|
bridge->id, rc);
|
|
}
|
|
|
|
static void dp_bridge_disable(struct drm_bridge *drm_bridge)
|
|
{
|
|
int rc = 0;
|
|
struct dp_bridge *bridge;
|
|
struct dp_display *dp;
|
|
|
|
if (!drm_bridge) {
|
|
pr_err("Invalid params\n");
|
|
return;
|
|
}
|
|
|
|
bridge = to_dp_bridge(drm_bridge);
|
|
if (!bridge->connector) {
|
|
pr_err("Invalid connector\n");
|
|
return;
|
|
}
|
|
|
|
if (!bridge->dp_panel) {
|
|
pr_err("Invalid dp_panel\n");
|
|
return;
|
|
}
|
|
|
|
dp = bridge->display;
|
|
|
|
if (!dp) {
|
|
pr_err("dp is null\n");
|
|
return;
|
|
}
|
|
|
|
if (dp)
|
|
sde_connector_helper_bridge_disable(bridge->connector);
|
|
|
|
rc = dp->pre_disable(dp, bridge->dp_panel);
|
|
if (rc) {
|
|
pr_err("[%d] DP display pre disable failed, rc=%d\n",
|
|
bridge->id, rc);
|
|
}
|
|
}
|
|
|
|
static void dp_bridge_post_disable(struct drm_bridge *drm_bridge)
|
|
{
|
|
int rc = 0;
|
|
struct dp_bridge *bridge;
|
|
struct dp_display *dp;
|
|
|
|
if (!drm_bridge) {
|
|
pr_err("Invalid params\n");
|
|
return;
|
|
}
|
|
|
|
bridge = to_dp_bridge(drm_bridge);
|
|
if (!bridge->connector) {
|
|
pr_err("Invalid connector\n");
|
|
return;
|
|
}
|
|
|
|
if (!bridge->dp_panel) {
|
|
pr_err("Invalid dp_panel\n");
|
|
return;
|
|
}
|
|
|
|
dp = bridge->display;
|
|
|
|
rc = dp->disable(dp, bridge->dp_panel);
|
|
if (rc) {
|
|
pr_err("[%d] DP display disable failed, rc=%d\n",
|
|
bridge->id, rc);
|
|
return;
|
|
}
|
|
|
|
rc = dp->unprepare(dp, bridge->dp_panel);
|
|
if (rc) {
|
|
pr_err("[%d] DP display unprepare failed, rc=%d\n",
|
|
bridge->id, rc);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void dp_bridge_mode_set(struct drm_bridge *drm_bridge,
|
|
struct drm_display_mode *mode,
|
|
struct drm_display_mode *adjusted_mode)
|
|
{
|
|
struct dp_bridge *bridge;
|
|
struct dp_display *dp;
|
|
|
|
if (!drm_bridge || !mode || !adjusted_mode) {
|
|
pr_err("Invalid params\n");
|
|
return;
|
|
}
|
|
|
|
bridge = to_dp_bridge(drm_bridge);
|
|
if (!bridge->connector) {
|
|
pr_err("Invalid connector\n");
|
|
return;
|
|
}
|
|
|
|
if (!bridge->dp_panel) {
|
|
pr_err("Invalid dp_panel\n");
|
|
return;
|
|
}
|
|
|
|
dp = bridge->display;
|
|
|
|
dp->convert_to_dp_mode(dp, bridge->dp_panel, adjusted_mode,
|
|
&bridge->dp_mode);
|
|
}
|
|
|
|
static bool dp_bridge_mode_fixup(struct drm_bridge *drm_bridge,
|
|
const struct drm_display_mode *mode,
|
|
struct drm_display_mode *adjusted_mode)
|
|
{
|
|
bool ret = true;
|
|
struct dp_display_mode dp_mode;
|
|
struct dp_bridge *bridge;
|
|
struct dp_display *dp;
|
|
|
|
if (!drm_bridge || !mode || !adjusted_mode) {
|
|
pr_err("Invalid params\n");
|
|
ret = false;
|
|
goto end;
|
|
}
|
|
|
|
bridge = to_dp_bridge(drm_bridge);
|
|
if (!bridge->connector) {
|
|
pr_err("Invalid connector\n");
|
|
ret = false;
|
|
goto end;
|
|
}
|
|
|
|
if (!bridge->dp_panel) {
|
|
pr_err("Invalid dp_panel\n");
|
|
ret = false;
|
|
goto end;
|
|
}
|
|
|
|
dp = bridge->display;
|
|
|
|
dp->convert_to_dp_mode(dp, bridge->dp_panel, mode, &dp_mode);
|
|
convert_to_drm_mode(&dp_mode, adjusted_mode);
|
|
end:
|
|
return ret;
|
|
}
|
|
|
|
static const struct drm_bridge_funcs dp_bridge_ops = {
|
|
.attach = dp_bridge_attach,
|
|
.mode_fixup = dp_bridge_mode_fixup,
|
|
.pre_enable = dp_bridge_pre_enable,
|
|
.enable = dp_bridge_enable,
|
|
.disable = dp_bridge_disable,
|
|
.post_disable = dp_bridge_post_disable,
|
|
.mode_set = dp_bridge_mode_set,
|
|
};
|
|
|
|
int dp_connector_config_hdr(struct drm_connector *connector, void *display,
|
|
struct sde_connector_state *c_state)
|
|
{
|
|
struct dp_display *dp = display;
|
|
struct sde_connector *sde_conn;
|
|
|
|
if (!display || !c_state || !connector) {
|
|
pr_err("invalid params\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
sde_conn = to_sde_connector(connector);
|
|
if (!sde_conn->drv_panel) {
|
|
pr_err("invalid dp panel\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
return dp->config_hdr(dp, sde_conn->drv_panel, &c_state->hdr_meta);
|
|
}
|
|
|
|
int dp_connector_post_init(struct drm_connector *connector, void *display)
|
|
{
|
|
int rc;
|
|
struct dp_display *dp_display = display;
|
|
struct sde_connector *sde_conn;
|
|
|
|
if (!dp_display || !connector)
|
|
return -EINVAL;
|
|
|
|
dp_display->base_connector = connector;
|
|
dp_display->bridge->connector = connector;
|
|
|
|
if (dp_display->post_init) {
|
|
rc = dp_display->post_init(dp_display);
|
|
if (rc)
|
|
goto end;
|
|
}
|
|
|
|
sde_conn = to_sde_connector(connector);
|
|
dp_display->bridge->dp_panel = sde_conn->drv_panel;
|
|
|
|
rc = dp_mst_init(dp_display);
|
|
end:
|
|
return rc;
|
|
}
|
|
|
|
int dp_connector_get_mode_info(struct drm_connector *connector,
|
|
const struct drm_display_mode *drm_mode,
|
|
struct msm_mode_info *mode_info,
|
|
u32 max_mixer_width, void *display)
|
|
{
|
|
const u32 single_intf = 1;
|
|
const u32 no_enc = 0;
|
|
struct msm_display_topology *topology;
|
|
struct sde_connector *sde_conn;
|
|
struct dp_panel *dp_panel;
|
|
struct dp_display_mode dp_mode;
|
|
struct dp_display *dp_disp = display;
|
|
struct msm_drm_private *priv;
|
|
int rc = 0;
|
|
|
|
if (!drm_mode || !mode_info || !max_mixer_width || !connector ||
|
|
!display) {
|
|
pr_err("invalid params\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
memset(mode_info, 0, sizeof(*mode_info));
|
|
|
|
sde_conn = to_sde_connector(connector);
|
|
dp_panel = sde_conn->drv_panel;
|
|
priv = connector->dev->dev_private;
|
|
|
|
topology = &mode_info->topology;
|
|
|
|
rc = msm_get_mixer_count(priv, drm_mode, max_mixer_width,
|
|
&topology->num_lm);
|
|
if (rc) {
|
|
pr_err("error getting mixer count, rc:%d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
topology->num_enc = no_enc;
|
|
topology->num_intf = single_intf;
|
|
|
|
mode_info->frame_rate = drm_mode->vrefresh;
|
|
mode_info->vtotal = drm_mode->vtotal;
|
|
|
|
mode_info->wide_bus_en = dp_panel->widebus_en;
|
|
|
|
dp_disp->convert_to_dp_mode(dp_disp, dp_panel, drm_mode, &dp_mode);
|
|
|
|
if (dp_mode.timing.comp_info.comp_ratio) {
|
|
memcpy(&mode_info->comp_info,
|
|
&dp_mode.timing.comp_info,
|
|
sizeof(mode_info->comp_info));
|
|
|
|
topology->num_enc = topology->num_lm;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int dp_connector_get_info(struct drm_connector *connector,
|
|
struct msm_display_info *info, void *data)
|
|
{
|
|
struct dp_display *display = data;
|
|
|
|
if (!info || !display || !display->drm_dev) {
|
|
pr_err("invalid params\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
info->intf_type = DRM_MODE_CONNECTOR_DisplayPort;
|
|
|
|
info->num_of_h_tiles = 1;
|
|
info->h_tile_instance[0] = 0;
|
|
info->is_connected = display->is_sst_connected;
|
|
info->capabilities = MSM_DISPLAY_CAP_VID_MODE | MSM_DISPLAY_CAP_EDID |
|
|
MSM_DISPLAY_CAP_HOT_PLUG;
|
|
|
|
return 0;
|
|
}
|
|
|
|
enum drm_connector_status dp_connector_detect(struct drm_connector *conn,
|
|
bool force,
|
|
void *display)
|
|
{
|
|
enum drm_connector_status status = connector_status_unknown;
|
|
struct msm_display_info info;
|
|
int rc;
|
|
|
|
if (!conn || !display)
|
|
return status;
|
|
|
|
/* get display dp_info */
|
|
memset(&info, 0x0, sizeof(info));
|
|
rc = dp_connector_get_info(conn, &info, display);
|
|
if (rc) {
|
|
pr_err("failed to get display info, rc=%d\n", rc);
|
|
return connector_status_disconnected;
|
|
}
|
|
|
|
if (info.capabilities & MSM_DISPLAY_CAP_HOT_PLUG)
|
|
status = (info.is_connected ? connector_status_connected :
|
|
connector_status_disconnected);
|
|
else
|
|
status = connector_status_connected;
|
|
|
|
conn->display_info.width_mm = info.width_mm;
|
|
conn->display_info.height_mm = info.height_mm;
|
|
|
|
return status;
|
|
}
|
|
|
|
void dp_connector_post_open(struct drm_connector *connector, void *display)
|
|
{
|
|
struct dp_display *dp;
|
|
|
|
if (!display) {
|
|
pr_err("invalid input\n");
|
|
return;
|
|
}
|
|
|
|
dp = display;
|
|
|
|
if (dp->post_open)
|
|
dp->post_open(dp);
|
|
}
|
|
|
|
int dp_connector_get_modes(struct drm_connector *connector,
|
|
void *display)
|
|
{
|
|
int rc = 0;
|
|
struct dp_display *dp;
|
|
struct dp_display_mode *dp_mode = NULL;
|
|
struct drm_display_mode *m, drm_mode;
|
|
struct sde_connector *sde_conn;
|
|
|
|
if (!connector || !display)
|
|
return 0;
|
|
|
|
sde_conn = to_sde_connector(connector);
|
|
if (!sde_conn->drv_panel) {
|
|
pr_err("invalid dp panel\n");
|
|
return 0;
|
|
}
|
|
|
|
dp = display;
|
|
|
|
dp_mode = kzalloc(sizeof(*dp_mode), GFP_KERNEL);
|
|
if (!dp_mode)
|
|
return 0;
|
|
|
|
/* pluggable case assumes EDID is read when HPD */
|
|
if (dp->is_sst_connected) {
|
|
rc = dp->get_modes(dp, sde_conn->drv_panel, dp_mode);
|
|
if (!rc)
|
|
pr_err("failed to get DP sink modes, rc=%d\n", rc);
|
|
|
|
if (dp_mode->timing.pixel_clk_khz) { /* valid DP mode */
|
|
memset(&drm_mode, 0x0, sizeof(drm_mode));
|
|
convert_to_drm_mode(dp_mode, &drm_mode);
|
|
m = drm_mode_duplicate(connector->dev, &drm_mode);
|
|
if (!m) {
|
|
pr_err("failed to add mode %ux%u\n",
|
|
drm_mode.hdisplay,
|
|
drm_mode.vdisplay);
|
|
kfree(dp_mode);
|
|
return 0;
|
|
}
|
|
m->width_mm = connector->display_info.width_mm;
|
|
m->height_mm = connector->display_info.height_mm;
|
|
drm_mode_probed_add(connector, m);
|
|
}
|
|
} else {
|
|
pr_err("No sink connected\n");
|
|
}
|
|
kfree(dp_mode);
|
|
|
|
return rc;
|
|
}
|
|
|
|
int dp_connnector_set_info_blob(struct drm_connector *connector,
|
|
void *info, void *display, struct msm_mode_info *mode_info)
|
|
{
|
|
struct dp_display *dp_display = display;
|
|
const char *display_type = NULL;
|
|
|
|
dp_display->get_display_type(dp_display, &display_type);
|
|
sde_kms_info_add_keystr(info,
|
|
"display type", display_type);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int dp_drm_bridge_init(void *data, struct drm_encoder *encoder)
|
|
{
|
|
int rc = 0;
|
|
struct dp_bridge *bridge;
|
|
struct drm_device *dev;
|
|
struct dp_display *display = data;
|
|
struct msm_drm_private *priv = NULL;
|
|
|
|
bridge = kzalloc(sizeof(*bridge), GFP_KERNEL);
|
|
if (!bridge) {
|
|
rc = -ENOMEM;
|
|
goto error;
|
|
}
|
|
|
|
dev = display->drm_dev;
|
|
bridge->display = display;
|
|
bridge->base.funcs = &dp_bridge_ops;
|
|
bridge->base.encoder = encoder;
|
|
|
|
priv = dev->dev_private;
|
|
|
|
rc = drm_bridge_attach(encoder, &bridge->base, NULL);
|
|
if (rc) {
|
|
pr_err("failed to attach bridge, rc=%d\n", rc);
|
|
goto error_free_bridge;
|
|
}
|
|
|
|
rc = display->request_irq(display);
|
|
if (rc) {
|
|
pr_err("request_irq failed, rc=%d\n", rc);
|
|
goto error_free_bridge;
|
|
}
|
|
|
|
encoder->bridge = &bridge->base;
|
|
priv->bridges[priv->num_bridges++] = &bridge->base;
|
|
display->bridge = bridge;
|
|
|
|
return 0;
|
|
error_free_bridge:
|
|
kfree(bridge);
|
|
error:
|
|
return rc;
|
|
}
|
|
|
|
void dp_drm_bridge_deinit(void *data)
|
|
{
|
|
struct dp_display *display = data;
|
|
struct dp_bridge *bridge = display->bridge;
|
|
|
|
if (bridge && bridge->base.encoder)
|
|
bridge->base.encoder->bridge = NULL;
|
|
|
|
kfree(bridge);
|
|
}
|
|
|
|
enum drm_mode_status dp_connector_mode_valid(struct drm_connector *connector,
|
|
struct drm_display_mode *mode, void *display)
|
|
{
|
|
struct dp_display *dp_disp;
|
|
struct sde_connector *sde_conn;
|
|
|
|
if (!mode || !display || !connector) {
|
|
pr_err("invalid params\n");
|
|
return MODE_ERROR;
|
|
}
|
|
|
|
sde_conn = to_sde_connector(connector);
|
|
if (!sde_conn->drv_panel) {
|
|
pr_err("invalid dp panel\n");
|
|
return MODE_ERROR;
|
|
}
|
|
|
|
dp_disp = display;
|
|
mode->vrefresh = drm_mode_vrefresh(mode);
|
|
|
|
return dp_disp->validate_mode(dp_disp, sde_conn->drv_panel, mode);
|
|
}
|
|
|
|
int dp_connector_update_pps(struct drm_connector *connector,
|
|
char *pps_cmd, void *display)
|
|
{
|
|
struct dp_display *dp_disp;
|
|
struct sde_connector *sde_conn;
|
|
|
|
if (!display || !connector) {
|
|
pr_err("invalid params\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
sde_conn = to_sde_connector(connector);
|
|
if (!sde_conn->drv_panel) {
|
|
pr_err("invalid dp panel\n");
|
|
return MODE_ERROR;
|
|
}
|
|
|
|
dp_disp = display;
|
|
return dp_disp->update_pps(dp_disp, connector, pps_cmd);
|
|
}
|
|
|