Commit 7731cb65 authored by Dave Airlie's avatar Dave Airlie

Merge tag 'drm-msm-fixes-2023-10-07' of https://gitlab.freedesktop.org/drm/msm into drm-fixes

Fixes for v6.6-rc5

- fix to not reset the PHY everytime we start link training but only
  do it if link training fails. Without this, the PLL unlocked
  interrupt fires causing "Unexpected DP AUX IRQ 0x01000000 when
  not busy" spam in the logs since last 2-3 cycles
- correct the highest bank bit to match downstream device tree for
  msm8998
- skip the video mode wait if the timing engine is not enabled. This
  was introduced after pre_enable flag for DSI video mode panels
  where we would end up waiting for the video mode done interrupt
  even before enabling timing engine causing error spam and long
  bootup times.
- check the correct return code of irq_of_parse_and_map() in DSI code
- avoid overflow issues in the dpu bandwidth calculation . This was
  exposed for high resolution displays and a critical fix to avoid
  atomic_check failure
- minor fix to add new lines in DP print messages.
- Fix to fail atomic_check() if the resolution exceeds max mdp clk.
  This leads to underflow otherwise if we try to allow that frame.
Signed-off-by: default avatarDave Airlie <airlied@redhat.com>
From: Rob Clark <robdclark@gmail.com>
Link: https://patchwork.freedesktop.org/patch/msgid/CAF6AEGv-HNxQ=VBtZ8geGzYJum9jtManEdbvhcjo_WWF_J9Ziw@mail.gmail.com
parents dcad98b1 10f20628
...@@ -119,6 +119,7 @@ static u64 _dpu_plane_calc_bw(const struct dpu_mdss_cfg *catalog, ...@@ -119,6 +119,7 @@ static u64 _dpu_plane_calc_bw(const struct dpu_mdss_cfg *catalog,
struct dpu_sw_pipe_cfg *pipe_cfg) struct dpu_sw_pipe_cfg *pipe_cfg)
{ {
int src_width, src_height, dst_height, fps; int src_width, src_height, dst_height, fps;
u64 plane_pixel_rate, plane_bit_rate;
u64 plane_prefill_bw; u64 plane_prefill_bw;
u64 plane_bw; u64 plane_bw;
u32 hw_latency_lines; u32 hw_latency_lines;
...@@ -136,13 +137,12 @@ static u64 _dpu_plane_calc_bw(const struct dpu_mdss_cfg *catalog, ...@@ -136,13 +137,12 @@ static u64 _dpu_plane_calc_bw(const struct dpu_mdss_cfg *catalog,
scale_factor = src_height > dst_height ? scale_factor = src_height > dst_height ?
mult_frac(src_height, 1, dst_height) : 1; mult_frac(src_height, 1, dst_height) : 1;
plane_bw = plane_pixel_rate = src_width * mode->vtotal * fps;
src_width * mode->vtotal * fps * fmt->bpp * plane_bit_rate = plane_pixel_rate * fmt->bpp;
scale_factor;
plane_prefill_bw = plane_bw = plane_bit_rate * scale_factor;
src_width * hw_latency_lines * fps * fmt->bpp *
scale_factor * mode->vtotal; plane_prefill_bw = plane_bw * hw_latency_lines;
if ((vbp+vpw) > hw_latency_lines) if ((vbp+vpw) > hw_latency_lines)
do_div(plane_prefill_bw, (vbp+vpw)); do_div(plane_prefill_bw, (vbp+vpw));
...@@ -733,9 +733,11 @@ static int dpu_plane_check_inline_rotation(struct dpu_plane *pdpu, ...@@ -733,9 +733,11 @@ static int dpu_plane_check_inline_rotation(struct dpu_plane *pdpu,
static int dpu_plane_atomic_check_pipe(struct dpu_plane *pdpu, static int dpu_plane_atomic_check_pipe(struct dpu_plane *pdpu,
struct dpu_sw_pipe *pipe, struct dpu_sw_pipe *pipe,
struct dpu_sw_pipe_cfg *pipe_cfg, struct dpu_sw_pipe_cfg *pipe_cfg,
const struct dpu_format *fmt) const struct dpu_format *fmt,
const struct drm_display_mode *mode)
{ {
uint32_t min_src_size; uint32_t min_src_size;
struct dpu_kms *kms = _dpu_plane_get_kms(&pdpu->base);
min_src_size = DPU_FORMAT_IS_YUV(fmt) ? 2 : 1; min_src_size = DPU_FORMAT_IS_YUV(fmt) ? 2 : 1;
...@@ -774,6 +776,12 @@ static int dpu_plane_atomic_check_pipe(struct dpu_plane *pdpu, ...@@ -774,6 +776,12 @@ static int dpu_plane_atomic_check_pipe(struct dpu_plane *pdpu,
return -EINVAL; return -EINVAL;
} }
/* max clk check */
if (_dpu_plane_calc_clk(mode, pipe_cfg) > kms->perf.max_core_clk_rate) {
DPU_DEBUG_PLANE(pdpu, "plane exceeds max mdp core clk limits\n");
return -E2BIG;
}
return 0; return 0;
} }
...@@ -899,12 +907,13 @@ static int dpu_plane_atomic_check(struct drm_plane *plane, ...@@ -899,12 +907,13 @@ static int dpu_plane_atomic_check(struct drm_plane *plane,
r_pipe_cfg->dst_rect.x1 = pipe_cfg->dst_rect.x2; r_pipe_cfg->dst_rect.x1 = pipe_cfg->dst_rect.x2;
} }
ret = dpu_plane_atomic_check_pipe(pdpu, pipe, pipe_cfg, fmt); ret = dpu_plane_atomic_check_pipe(pdpu, pipe, pipe_cfg, fmt, &crtc_state->adjusted_mode);
if (ret) if (ret)
return ret; return ret;
if (r_pipe->sspp) { if (r_pipe->sspp) {
ret = dpu_plane_atomic_check_pipe(pdpu, r_pipe, r_pipe_cfg, fmt); ret = dpu_plane_atomic_check_pipe(pdpu, r_pipe, r_pipe_cfg, fmt,
&crtc_state->adjusted_mode);
if (ret) if (ret)
return ret; return ret;
} }
......
...@@ -1774,13 +1774,6 @@ int dp_ctrl_on_link(struct dp_ctrl *dp_ctrl) ...@@ -1774,13 +1774,6 @@ int dp_ctrl_on_link(struct dp_ctrl *dp_ctrl)
return rc; return rc;
while (--link_train_max_retries) { while (--link_train_max_retries) {
rc = dp_ctrl_reinitialize_mainlink(ctrl);
if (rc) {
DRM_ERROR("Failed to reinitialize mainlink. rc=%d\n",
rc);
break;
}
training_step = DP_TRAINING_NONE; training_step = DP_TRAINING_NONE;
rc = dp_ctrl_setup_main_link(ctrl, &training_step); rc = dp_ctrl_setup_main_link(ctrl, &training_step);
if (rc == 0) { if (rc == 0) {
...@@ -1832,6 +1825,12 @@ int dp_ctrl_on_link(struct dp_ctrl *dp_ctrl) ...@@ -1832,6 +1825,12 @@ int dp_ctrl_on_link(struct dp_ctrl *dp_ctrl)
/* stop link training before start re training */ /* stop link training before start re training */
dp_ctrl_clear_training_pattern(ctrl); dp_ctrl_clear_training_pattern(ctrl);
} }
rc = dp_ctrl_reinitialize_mainlink(ctrl);
if (rc) {
DRM_ERROR("Failed to reinitialize mainlink. rc=%d\n", rc);
break;
}
} }
if (ctrl->link->sink_request & DP_TEST_LINK_PHY_TEST_PATTERN) if (ctrl->link->sink_request & DP_TEST_LINK_PHY_TEST_PATTERN)
......
...@@ -1090,7 +1090,7 @@ int dp_link_process_request(struct dp_link *dp_link) ...@@ -1090,7 +1090,7 @@ int dp_link_process_request(struct dp_link *dp_link)
} else if (dp_link_read_psr_error_status(link)) { } else if (dp_link_read_psr_error_status(link)) {
DRM_ERROR("PSR IRQ_HPD received\n"); DRM_ERROR("PSR IRQ_HPD received\n");
} else if (dp_link_psr_capability_changed(link)) { } else if (dp_link_psr_capability_changed(link)) {
drm_dbg_dp(link->drm_dev, "PSR Capability changed"); drm_dbg_dp(link->drm_dev, "PSR Capability changed\n");
} else { } else {
ret = dp_link_process_link_status_update(link); ret = dp_link_process_link_status_update(link);
if (!ret) { if (!ret) {
...@@ -1107,7 +1107,7 @@ int dp_link_process_request(struct dp_link *dp_link) ...@@ -1107,7 +1107,7 @@ int dp_link_process_request(struct dp_link *dp_link)
} }
} }
drm_dbg_dp(link->drm_dev, "sink request=%#x", drm_dbg_dp(link->drm_dev, "sink request=%#x\n",
dp_link->sink_request); dp_link->sink_request);
return ret; return ret;
} }
......
...@@ -1082,9 +1082,21 @@ static void dsi_wait4video_done(struct msm_dsi_host *msm_host) ...@@ -1082,9 +1082,21 @@ static void dsi_wait4video_done(struct msm_dsi_host *msm_host)
static void dsi_wait4video_eng_busy(struct msm_dsi_host *msm_host) static void dsi_wait4video_eng_busy(struct msm_dsi_host *msm_host)
{ {
u32 data;
if (!(msm_host->mode_flags & MIPI_DSI_MODE_VIDEO)) if (!(msm_host->mode_flags & MIPI_DSI_MODE_VIDEO))
return; return;
data = dsi_read(msm_host, REG_DSI_STATUS0);
/* if video mode engine is not busy, its because
* either timing engine was not turned on or the
* DSI controller has finished transmitting the video
* data already, so no need to wait in those cases
*/
if (!(data & DSI_STATUS0_VIDEO_MODE_ENGINE_BUSY))
return;
if (msm_host->power_on && msm_host->enabled) { if (msm_host->power_on && msm_host->enabled) {
dsi_wait4video_done(msm_host); dsi_wait4video_done(msm_host);
/* delay 4 ms to skip BLLP */ /* delay 4 ms to skip BLLP */
...@@ -1894,10 +1906,9 @@ int msm_dsi_host_init(struct msm_dsi *msm_dsi) ...@@ -1894,10 +1906,9 @@ int msm_dsi_host_init(struct msm_dsi *msm_dsi)
} }
msm_host->irq = irq_of_parse_and_map(pdev->dev.of_node, 0); msm_host->irq = irq_of_parse_and_map(pdev->dev.of_node, 0);
if (msm_host->irq < 0) { if (!msm_host->irq) {
ret = msm_host->irq; dev_err(&pdev->dev, "failed to get irq\n");
dev_err(&pdev->dev, "failed to get irq: %d\n", ret); return -EINVAL;
return ret;
} }
/* do not autoenable, will be enabled later */ /* do not autoenable, will be enabled later */
......
...@@ -511,7 +511,7 @@ static int mdss_remove(struct platform_device *pdev) ...@@ -511,7 +511,7 @@ static int mdss_remove(struct platform_device *pdev)
static const struct msm_mdss_data msm8998_data = { static const struct msm_mdss_data msm8998_data = {
.ubwc_enc_version = UBWC_1_0, .ubwc_enc_version = UBWC_1_0,
.ubwc_dec_version = UBWC_1_0, .ubwc_dec_version = UBWC_1_0,
.highest_bank_bit = 1, .highest_bank_bit = 2,
}; };
static const struct msm_mdss_data qcm2290_data = { static const struct msm_mdss_data qcm2290_data = {
......
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