commit 3f65b1e2f424f44585bd701024a3bfd0b1e0ade2 upstream. Current DP driver implementation has adding safe mode done at dp_hpd_plug_handle() which is expected to be executed under event thread context. However there is possible circular locking happen (see blow stack trace) after edp driver call dp_hpd_plug_handle() from dp_bridge_enable() which is executed under drm_thread context. After review all possibilities methods and as discussed on https://patchwork.freedesktop.org/patch/483155/, supporting EDID compliance tests in the driver is quite hacky. As seen with other vendor drivers, supporting these will be much easier with IGT. Hence removing all the related fail safe code for it so that no possibility of circular lock will happen. Reviewed-by: Stephen Boyd <swboyd@chromium.org> Reviewed-by: Douglas Anderson <dianders@chromium.org> Reviewed-by: Dmitry Baryshkov <dmitry.baryshkov@linaro.org> ====================================================== WARNING: possible circular locking dependency detected 5.15.35-lockdep #6 Tainted: G W ------------------------------------------------------ frecon/429 is trying to acquire lock: ffffff808dc3c4e8 (&dev->mode_config.mutex){+.+.}-{3:3}, at: dp_panel_add_fail_safe_mode+0x4c/0xa0 but task is already holding lock: ffffff808dc441e0 (&kms->commit_lock[i]){+.+.}-{3:3}, at: lock_crtcs+0xb4/0x124 which lock already depends on the new lock. the existing dependency chain (in reverse order) is: -> #3 (&kms->commit_lock[i]){+.+.}-{3:3}: __mutex_lock_common+0x174/0x1a64 mutex_lock_nested+0x98/0xac lock_crtcs+0xb4/0x124 msm_atomic_commit_tail+0x330/0x748 commit_tail+0x19c/0x278 drm_atomic_helper_commit+0x1dc/0x1f0 drm_atomic_commit+0xc0/0xd8 drm_atomic_helper_set_config+0xb4/0x134 drm_mode_setcrtc+0x688/0x1248 drm_ioctl_kernel+0x1e4/0x338 drm_ioctl+0x3a4/0x684 __arm64_sys_ioctl+0x118/0x154 invoke_syscall+0x78/0x224 el0_svc_common+0x178/0x200 do_el0_svc+0x94/0x13c el0_svc+0x5c/0xec el0t_64_sync_handler+0x78/0x108 el0t_64_sync+0x1a4/0x1a8 -> #2 (crtc_ww_class_mutex){+.+.}-{3:3}: __mutex_lock_common+0x174/0x1a64 ww_mutex_lock+0xb8/0x278 modeset_lock+0x304/0x4ac drm_modeset_lock+0x4c/0x7c drmm_mode_config_init+0x4a8/0xc50 msm_drm_init+0x274/0xac0 msm_drm_bind+0x20/0x2c try_to_bring_up_master+0x3dc/0x470 __component_add+0x18c/0x3c0 component_add+0x1c/0x28 dp_display_probe+0x954/0xa98 platform_probe+0x124/0x15c really_probe+0x1b0/0x5f8 __driver_probe_device+0x174/0x20c driver_probe_device+0x70/0x134 __device_attach_driver+0x130/0x1d0 bus_for_each_drv+0xfc/0x14c __device_attach+0x1bc/0x2bc device_initial_probe+0x1c/0x28 bus_probe_device+0x94/0x178 deferred_probe_work_func+0x1a4/0x1f0 process_one_work+0x5d4/0x9dc worker_thread+0x898/0xccc kthread+0x2d4/0x3d4 ret_from_fork+0x10/0x20 -> #1 (crtc_ww_class_acquire){+.+.}-{0:0}: ww_acquire_init+0x1c4/0x2c8 drm_modeset_acquire_init+0x44/0xc8 drm_helper_probe_single_connector_modes+0xb0/0x12dc drm_mode_getconnector+0x5dc/0xfe8 drm_ioctl_kernel+0x1e4/0x338 drm_ioctl+0x3a4/0x684 __arm64_sys_ioctl+0x118/0x154 invoke_syscall+0x78/0x224 el0_svc_common+0x178/0x200 do_el0_svc+0x94/0x13c el0_svc+0x5c/0xec el0t_64_sync_handler+0x78/0x108 el0t_64_sync+0x1a4/0x1a8 -> #0 (&dev->mode_config.mutex){+.+.}-{3:3}: __lock_acquire+0x2650/0x672c lock_acquire+0x1b4/0x4ac __mutex_lock_common+0x174/0x1a64 mutex_lock_nested+0x98/0xac dp_panel_add_fail_safe_mode+0x4c/0xa0 dp_hpd_plug_handle+0x1f0/0x280 dp_bridge_enable+0x94/0x2b8 drm_atomic_bridge_chain_enable+0x11c/0x168 drm_atomic_helper_commit_modeset_enables+0x500/0x740 msm_atomic_commit_tail+0x3e4/0x748 commit_tail+0x19c/0x278 drm_atomic_helper_commit+0x1dc/0x1f0 drm_atomic_commit+0xc0/0xd8 drm_atomic_helper_set_config+0xb4/0x134 drm_mode_setcrtc+0x688/0x1248 drm_ioctl_kernel+0x1e4/0x338 drm_ioctl+0x3a4/0x684 __arm64_sys_ioctl+0x118/0x154 invoke_syscall+0x78/0x224 el0_svc_common+0x178/0x200 do_el0_svc+0x94/0x13c el0_svc+0x5c/0xec el0t_64_sync_handler+0x78/0x108 el0t_64_sync+0x1a4/0x1a8 Changes in v2: -- re text commit title -- remove all fail safe mode Changes in v3: -- remove dp_panel_add_fail_safe_mode() from dp_panel.h -- add Fixes Changes in v5: -- to=dianders@chromium.org Changes in v6: -- fix Fixes commit ID Fixes: 8b2c181e3dcf ("drm/msm/dp: add fail safe mode outside of event_mutex context") Reported-by: Douglas Anderson <dianders@chromium.org> Signed-off-by: Kuogee Hsieh <quic_khsieh@quicinc.com> Link: https://lore.kernel.org/r/1651007534-31842-1-git-send-email-quic_khsieh@quicinc.com Signed-off-by: Rob Clark <robdclark@chromium.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
471 lines
11 KiB
C
471 lines
11 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (c) 2012-2020, The Linux Foundation. All rights reserved.
|
|
*/
|
|
|
|
#include "dp_panel.h"
|
|
|
|
#include <drm/drm_connector.h>
|
|
#include <drm/drm_edid.h>
|
|
#include <drm/drm_print.h>
|
|
|
|
struct dp_panel_private {
|
|
struct device *dev;
|
|
struct dp_panel dp_panel;
|
|
struct drm_dp_aux *aux;
|
|
struct dp_link *link;
|
|
struct dp_catalog *catalog;
|
|
bool panel_on;
|
|
bool aux_cfg_update_done;
|
|
};
|
|
|
|
static int dp_panel_read_dpcd(struct dp_panel *dp_panel)
|
|
{
|
|
int rc = 0;
|
|
size_t len;
|
|
ssize_t rlen;
|
|
struct dp_panel_private *panel;
|
|
struct dp_link_info *link_info;
|
|
u8 *dpcd, major = 0, minor = 0, temp;
|
|
u32 offset = DP_DPCD_REV;
|
|
|
|
dpcd = dp_panel->dpcd;
|
|
|
|
panel = container_of(dp_panel, struct dp_panel_private, dp_panel);
|
|
link_info = &dp_panel->link_info;
|
|
|
|
rlen = drm_dp_dpcd_read(panel->aux, offset,
|
|
dpcd, (DP_RECEIVER_CAP_SIZE + 1));
|
|
if (rlen < (DP_RECEIVER_CAP_SIZE + 1)) {
|
|
DRM_ERROR("dpcd read failed, rlen=%zd\n", rlen);
|
|
if (rlen == -ETIMEDOUT)
|
|
rc = rlen;
|
|
else
|
|
rc = -EINVAL;
|
|
|
|
goto end;
|
|
}
|
|
|
|
temp = dpcd[DP_TRAINING_AUX_RD_INTERVAL];
|
|
|
|
/* check for EXTENDED_RECEIVER_CAPABILITY_FIELD_PRESENT */
|
|
if (temp & BIT(7)) {
|
|
DRM_DEBUG_DP("using EXTENDED_RECEIVER_CAPABILITY_FIELD\n");
|
|
offset = DPRX_EXTENDED_DPCD_FIELD;
|
|
}
|
|
|
|
rlen = drm_dp_dpcd_read(panel->aux, offset,
|
|
dpcd, (DP_RECEIVER_CAP_SIZE + 1));
|
|
if (rlen < (DP_RECEIVER_CAP_SIZE + 1)) {
|
|
DRM_ERROR("dpcd read failed, rlen=%zd\n", rlen);
|
|
if (rlen == -ETIMEDOUT)
|
|
rc = rlen;
|
|
else
|
|
rc = -EINVAL;
|
|
|
|
goto end;
|
|
}
|
|
|
|
link_info->revision = dpcd[DP_DPCD_REV];
|
|
major = (link_info->revision >> 4) & 0x0f;
|
|
minor = link_info->revision & 0x0f;
|
|
|
|
link_info->rate = drm_dp_bw_code_to_link_rate(dpcd[DP_MAX_LINK_RATE]);
|
|
link_info->num_lanes = dpcd[DP_MAX_LANE_COUNT] & DP_MAX_LANE_COUNT_MASK;
|
|
|
|
if (link_info->num_lanes > dp_panel->max_dp_lanes)
|
|
link_info->num_lanes = dp_panel->max_dp_lanes;
|
|
|
|
/* Limit support upto HBR2 until HBR3 support is added */
|
|
if (link_info->rate >= (drm_dp_bw_code_to_link_rate(DP_LINK_BW_5_4)))
|
|
link_info->rate = drm_dp_bw_code_to_link_rate(DP_LINK_BW_5_4);
|
|
|
|
DRM_DEBUG_DP("version: %d.%d\n", major, minor);
|
|
DRM_DEBUG_DP("link_rate=%d\n", link_info->rate);
|
|
DRM_DEBUG_DP("lane_count=%d\n", link_info->num_lanes);
|
|
|
|
if (drm_dp_enhanced_frame_cap(dpcd))
|
|
link_info->capabilities |= DP_LINK_CAP_ENHANCED_FRAMING;
|
|
|
|
dp_panel->dfp_present = dpcd[DP_DOWNSTREAMPORT_PRESENT];
|
|
dp_panel->dfp_present &= DP_DWN_STRM_PORT_PRESENT;
|
|
|
|
if (dp_panel->dfp_present && (dpcd[DP_DPCD_REV] > 0x10)) {
|
|
dp_panel->ds_port_cnt = dpcd[DP_DOWN_STREAM_PORT_COUNT];
|
|
dp_panel->ds_port_cnt &= DP_PORT_COUNT_MASK;
|
|
len = DP_DOWNSTREAM_PORTS * DP_DOWNSTREAM_CAP_SIZE;
|
|
|
|
rlen = drm_dp_dpcd_read(panel->aux,
|
|
DP_DOWNSTREAM_PORT_0, dp_panel->ds_cap_info, len);
|
|
if (rlen < len) {
|
|
DRM_ERROR("ds port status failed, rlen=%zd\n", rlen);
|
|
rc = -EINVAL;
|
|
goto end;
|
|
}
|
|
}
|
|
|
|
end:
|
|
return rc;
|
|
}
|
|
|
|
static u32 dp_panel_get_supported_bpp(struct dp_panel *dp_panel,
|
|
u32 mode_edid_bpp, u32 mode_pclk_khz)
|
|
{
|
|
struct dp_link_info *link_info;
|
|
const u32 max_supported_bpp = 30, min_supported_bpp = 18;
|
|
u32 bpp = 0, data_rate_khz = 0;
|
|
|
|
bpp = min_t(u32, mode_edid_bpp, max_supported_bpp);
|
|
|
|
link_info = &dp_panel->link_info;
|
|
data_rate_khz = link_info->num_lanes * link_info->rate * 8;
|
|
|
|
while (bpp > min_supported_bpp) {
|
|
if (mode_pclk_khz * bpp <= data_rate_khz)
|
|
break;
|
|
bpp -= 6;
|
|
}
|
|
|
|
return bpp;
|
|
}
|
|
|
|
static int dp_panel_update_modes(struct drm_connector *connector,
|
|
struct edid *edid)
|
|
{
|
|
int rc = 0;
|
|
|
|
if (edid) {
|
|
rc = drm_connector_update_edid_property(connector, edid);
|
|
if (rc) {
|
|
DRM_ERROR("failed to update edid property %d\n", rc);
|
|
return rc;
|
|
}
|
|
rc = drm_add_edid_modes(connector, edid);
|
|
return rc;
|
|
}
|
|
|
|
rc = drm_connector_update_edid_property(connector, NULL);
|
|
if (rc)
|
|
DRM_ERROR("failed to update edid property %d\n", rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
int dp_panel_read_sink_caps(struct dp_panel *dp_panel,
|
|
struct drm_connector *connector)
|
|
{
|
|
int rc = 0, bw_code;
|
|
int rlen, count;
|
|
struct dp_panel_private *panel;
|
|
|
|
if (!dp_panel || !connector) {
|
|
DRM_ERROR("invalid input\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
panel = container_of(dp_panel, struct dp_panel_private, dp_panel);
|
|
|
|
rc = dp_panel_read_dpcd(dp_panel);
|
|
if (rc) {
|
|
DRM_ERROR("read dpcd failed %d\n", rc);
|
|
return rc;
|
|
}
|
|
|
|
bw_code = drm_dp_link_rate_to_bw_code(dp_panel->link_info.rate);
|
|
if (!is_link_rate_valid(bw_code) ||
|
|
!is_lane_count_valid(dp_panel->link_info.num_lanes) ||
|
|
(bw_code > dp_panel->max_bw_code)) {
|
|
DRM_ERROR("Illegal link rate=%d lane=%d\n", dp_panel->link_info.rate,
|
|
dp_panel->link_info.num_lanes);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (dp_panel->dfp_present) {
|
|
rlen = drm_dp_dpcd_read(panel->aux, DP_SINK_COUNT,
|
|
&count, 1);
|
|
if (rlen == 1) {
|
|
count = DP_GET_SINK_COUNT(count);
|
|
if (!count) {
|
|
DRM_ERROR("no downstream ports connected\n");
|
|
panel->link->sink_count = 0;
|
|
rc = -ENOTCONN;
|
|
goto end;
|
|
}
|
|
}
|
|
}
|
|
|
|
kfree(dp_panel->edid);
|
|
dp_panel->edid = NULL;
|
|
|
|
dp_panel->edid = drm_get_edid(connector,
|
|
&panel->aux->ddc);
|
|
if (!dp_panel->edid) {
|
|
DRM_ERROR("panel edid read failed\n");
|
|
/* check edid read fail is due to unplug */
|
|
if (!dp_catalog_link_is_connected(panel->catalog)) {
|
|
rc = -ETIMEDOUT;
|
|
goto end;
|
|
}
|
|
}
|
|
|
|
if (panel->aux_cfg_update_done) {
|
|
DRM_DEBUG_DP("read DPCD with updated AUX config\n");
|
|
rc = dp_panel_read_dpcd(dp_panel);
|
|
bw_code = drm_dp_link_rate_to_bw_code(dp_panel->link_info.rate);
|
|
if (rc || !is_link_rate_valid(bw_code) ||
|
|
!is_lane_count_valid(dp_panel->link_info.num_lanes)
|
|
|| (bw_code > dp_panel->max_bw_code)) {
|
|
DRM_ERROR("read dpcd failed %d\n", rc);
|
|
return rc;
|
|
}
|
|
panel->aux_cfg_update_done = false;
|
|
}
|
|
end:
|
|
return rc;
|
|
}
|
|
|
|
u32 dp_panel_get_mode_bpp(struct dp_panel *dp_panel,
|
|
u32 mode_edid_bpp, u32 mode_pclk_khz)
|
|
{
|
|
struct dp_panel_private *panel;
|
|
u32 bpp = mode_edid_bpp;
|
|
|
|
if (!dp_panel || !mode_edid_bpp || !mode_pclk_khz) {
|
|
DRM_ERROR("invalid input\n");
|
|
return 0;
|
|
}
|
|
|
|
panel = container_of(dp_panel, struct dp_panel_private, dp_panel);
|
|
|
|
if (dp_panel->video_test)
|
|
bpp = dp_link_bit_depth_to_bpp(
|
|
panel->link->test_video.test_bit_depth);
|
|
else
|
|
bpp = dp_panel_get_supported_bpp(dp_panel, mode_edid_bpp,
|
|
mode_pclk_khz);
|
|
|
|
return bpp;
|
|
}
|
|
|
|
int dp_panel_get_modes(struct dp_panel *dp_panel,
|
|
struct drm_connector *connector, struct dp_display_mode *mode)
|
|
{
|
|
if (!dp_panel) {
|
|
DRM_ERROR("invalid input\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (dp_panel->edid)
|
|
return dp_panel_update_modes(connector, dp_panel->edid);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static u8 dp_panel_get_edid_checksum(struct edid *edid)
|
|
{
|
|
struct edid *last_block;
|
|
u8 *raw_edid;
|
|
bool is_edid_corrupt = false;
|
|
|
|
if (!edid) {
|
|
DRM_ERROR("invalid edid input\n");
|
|
return 0;
|
|
}
|
|
|
|
raw_edid = (u8 *)edid;
|
|
raw_edid += (edid->extensions * EDID_LENGTH);
|
|
last_block = (struct edid *)raw_edid;
|
|
|
|
/* block type extension */
|
|
drm_edid_block_valid(raw_edid, 1, false, &is_edid_corrupt);
|
|
if (!is_edid_corrupt)
|
|
return last_block->checksum;
|
|
|
|
DRM_ERROR("Invalid block, no checksum\n");
|
|
return 0;
|
|
}
|
|
|
|
void dp_panel_handle_sink_request(struct dp_panel *dp_panel)
|
|
{
|
|
struct dp_panel_private *panel;
|
|
|
|
if (!dp_panel) {
|
|
DRM_ERROR("invalid input\n");
|
|
return;
|
|
}
|
|
|
|
panel = container_of(dp_panel, struct dp_panel_private, dp_panel);
|
|
|
|
if (panel->link->sink_request & DP_TEST_LINK_EDID_READ) {
|
|
u8 checksum;
|
|
|
|
if (dp_panel->edid)
|
|
checksum = dp_panel_get_edid_checksum(dp_panel->edid);
|
|
else
|
|
checksum = dp_panel->connector->real_edid_checksum;
|
|
|
|
dp_link_send_edid_checksum(panel->link, checksum);
|
|
dp_link_send_test_response(panel->link);
|
|
}
|
|
}
|
|
|
|
void dp_panel_tpg_config(struct dp_panel *dp_panel, bool enable)
|
|
{
|
|
struct dp_catalog *catalog;
|
|
struct dp_panel_private *panel;
|
|
|
|
if (!dp_panel) {
|
|
DRM_ERROR("invalid input\n");
|
|
return;
|
|
}
|
|
|
|
panel = container_of(dp_panel, struct dp_panel_private, dp_panel);
|
|
catalog = panel->catalog;
|
|
|
|
if (!panel->panel_on) {
|
|
DRM_DEBUG_DP("DP panel not enabled, handle TPG on next on\n");
|
|
return;
|
|
}
|
|
|
|
if (!enable) {
|
|
dp_catalog_panel_tpg_disable(catalog);
|
|
return;
|
|
}
|
|
|
|
DRM_DEBUG_DP("%s: calling catalog tpg_enable\n", __func__);
|
|
dp_catalog_panel_tpg_enable(catalog, &panel->dp_panel.dp_mode.drm_mode);
|
|
}
|
|
|
|
void dp_panel_dump_regs(struct dp_panel *dp_panel)
|
|
{
|
|
struct dp_catalog *catalog;
|
|
struct dp_panel_private *panel;
|
|
|
|
panel = container_of(dp_panel, struct dp_panel_private, dp_panel);
|
|
catalog = panel->catalog;
|
|
|
|
dp_catalog_dump_regs(catalog);
|
|
}
|
|
|
|
int dp_panel_timing_cfg(struct dp_panel *dp_panel)
|
|
{
|
|
u32 data, total_ver, total_hor;
|
|
struct dp_catalog *catalog;
|
|
struct dp_panel_private *panel;
|
|
struct drm_display_mode *drm_mode;
|
|
|
|
panel = container_of(dp_panel, struct dp_panel_private, dp_panel);
|
|
catalog = panel->catalog;
|
|
drm_mode = &panel->dp_panel.dp_mode.drm_mode;
|
|
|
|
DRM_DEBUG_DP("width=%d hporch= %d %d %d\n",
|
|
drm_mode->hdisplay, drm_mode->htotal - drm_mode->hsync_end,
|
|
drm_mode->hsync_start - drm_mode->hdisplay,
|
|
drm_mode->hsync_end - drm_mode->hsync_start);
|
|
|
|
DRM_DEBUG_DP("height=%d vporch= %d %d %d\n",
|
|
drm_mode->vdisplay, drm_mode->vtotal - drm_mode->vsync_end,
|
|
drm_mode->vsync_start - drm_mode->vdisplay,
|
|
drm_mode->vsync_end - drm_mode->vsync_start);
|
|
|
|
total_hor = drm_mode->htotal;
|
|
|
|
total_ver = drm_mode->vtotal;
|
|
|
|
data = total_ver;
|
|
data <<= 16;
|
|
data |= total_hor;
|
|
|
|
catalog->total = data;
|
|
|
|
data = (drm_mode->vtotal - drm_mode->vsync_start);
|
|
data <<= 16;
|
|
data |= (drm_mode->htotal - drm_mode->hsync_start);
|
|
|
|
catalog->sync_start = data;
|
|
|
|
data = drm_mode->vsync_end - drm_mode->vsync_start;
|
|
data <<= 16;
|
|
data |= (panel->dp_panel.dp_mode.v_active_low << 31);
|
|
data |= drm_mode->hsync_end - drm_mode->hsync_start;
|
|
data |= (panel->dp_panel.dp_mode.h_active_low << 15);
|
|
|
|
catalog->width_blanking = data;
|
|
|
|
data = drm_mode->vdisplay;
|
|
data <<= 16;
|
|
data |= drm_mode->hdisplay;
|
|
|
|
catalog->dp_active = data;
|
|
|
|
dp_catalog_panel_timing_cfg(catalog);
|
|
panel->panel_on = true;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int dp_panel_init_panel_info(struct dp_panel *dp_panel)
|
|
{
|
|
struct drm_display_mode *drm_mode;
|
|
|
|
drm_mode = &dp_panel->dp_mode.drm_mode;
|
|
|
|
/*
|
|
* print resolution info as this is a result
|
|
* of user initiated action of cable connection
|
|
*/
|
|
DRM_DEBUG_DP("SET NEW RESOLUTION:\n");
|
|
DRM_DEBUG_DP("%dx%d@%dfps\n", drm_mode->hdisplay,
|
|
drm_mode->vdisplay, drm_mode_vrefresh(drm_mode));
|
|
DRM_DEBUG_DP("h_porches(back|front|width) = (%d|%d|%d)\n",
|
|
drm_mode->htotal - drm_mode->hsync_end,
|
|
drm_mode->hsync_start - drm_mode->hdisplay,
|
|
drm_mode->hsync_end - drm_mode->hsync_start);
|
|
DRM_DEBUG_DP("v_porches(back|front|width) = (%d|%d|%d)\n",
|
|
drm_mode->vtotal - drm_mode->vsync_end,
|
|
drm_mode->vsync_start - drm_mode->vdisplay,
|
|
drm_mode->vsync_end - drm_mode->vsync_start);
|
|
DRM_DEBUG_DP("pixel clock (KHz)=(%d)\n", drm_mode->clock);
|
|
DRM_DEBUG_DP("bpp = %d\n", dp_panel->dp_mode.bpp);
|
|
|
|
dp_panel->dp_mode.bpp = max_t(u32, 18,
|
|
min_t(u32, dp_panel->dp_mode.bpp, 30));
|
|
DRM_DEBUG_DP("updated bpp = %d\n", dp_panel->dp_mode.bpp);
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct dp_panel *dp_panel_get(struct dp_panel_in *in)
|
|
{
|
|
struct dp_panel_private *panel;
|
|
struct dp_panel *dp_panel;
|
|
|
|
if (!in->dev || !in->catalog || !in->aux || !in->link) {
|
|
DRM_ERROR("invalid input\n");
|
|
return ERR_PTR(-EINVAL);
|
|
}
|
|
|
|
panel = devm_kzalloc(in->dev, sizeof(*panel), GFP_KERNEL);
|
|
if (!panel)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
panel->dev = in->dev;
|
|
panel->aux = in->aux;
|
|
panel->catalog = in->catalog;
|
|
panel->link = in->link;
|
|
|
|
dp_panel = &panel->dp_panel;
|
|
dp_panel->max_bw_code = DP_LINK_BW_8_1;
|
|
panel->aux_cfg_update_done = false;
|
|
|
|
return dp_panel;
|
|
}
|
|
|
|
void dp_panel_put(struct dp_panel *dp_panel)
|
|
{
|
|
if (!dp_panel)
|
|
return;
|
|
|
|
kfree(dp_panel->edid);
|
|
}
|