forked from Minki/linux
- WARN if plane src coords are too big (Ville)
- Prevent double YUV range correction on HDR planes (Andres) - DP MST related Fixes (Sean, Imre) - More clean-up around DRAM detection code (Jose) - Actually async flips enable for all ilk+ platforms (Ville) -----BEGIN PGP SIGNATURE----- iQEzBAABCAAdFiEEbSBwaO7dZQkcLOKj+mJfZA7rE8oFAmAUkUIACgkQ+mJfZA7r E8rdqAgArg9X7LaZ9pnrPNDK2CSIZ7OSFmLjqYePDlKM+13TstM0NgMzMMCad7oM 0687oOaoT8NTaTvs5mQOG1TMWm8EfFsm1hzY8SRBDqHIjROUQWjYJRU202yWB2Ik Od5rsJ7U+RiVmdTCU0vPahSCSgqLS7s9/Cj0uxjP5q7Fpg8ab7R5pQL3a/XFqRFY VEynVt9GyGV/0bD7S8n6sJTKKMtdjWif1t6OiIdYZeU2UQY9juaIs+WuC6HaL+Zr mq2p1zUJfpEK1r7GupWUHQm/bhwhaBVbuqvhWZFFyyv5gyyHY8dLTXmAgZioMEJT F0TZR5zMCuBFJ0BSo7IuhJnJsBSHtQ== =VcVL -----END PGP SIGNATURE----- Merge tag 'drm-intel-next-2021-01-29' of git://anongit.freedesktop.org/drm/drm-intel into drm-next - WARN if plane src coords are too big (Ville) - Prevent double YUV range correction on HDR planes (Andres) - DP MST related Fixes (Sean, Imre) - More clean-up around DRAM detection code (Jose) - Actually async flips enable for all ilk+ platforms (Ville) Signed-off-by: Dave Airlie <airlied@redhat.com> From: Rodrigo Vivi <rodrigo.vivi@intel.com> Link: https://patchwork.freedesktop.org/patch/msgid/20210129225328.GA1041349@intel.com
This commit is contained in:
commit
ce7c3bded6
@ -3629,14 +3629,26 @@ static int drm_dp_send_up_ack_reply(struct drm_dp_mst_topology_mgr *mgr,
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int drm_dp_get_vc_payload_bw(u8 dp_link_bw, u8 dp_link_count)
|
||||
/**
|
||||
* drm_dp_get_vc_payload_bw - get the VC payload BW for an MST link
|
||||
* @link_rate: link rate in 10kbits/s units
|
||||
* @link_lane_count: lane count
|
||||
*
|
||||
* Calculate the total bandwidth of a MultiStream Transport link. The returned
|
||||
* value is in units of PBNs/(timeslots/1 MTP). This value can be used to
|
||||
* convert the number of PBNs required for a given stream to the number of
|
||||
* timeslots this stream requires in each MTP.
|
||||
*/
|
||||
int drm_dp_get_vc_payload_bw(int link_rate, int link_lane_count)
|
||||
{
|
||||
if (dp_link_bw == 0 || dp_link_count == 0)
|
||||
DRM_DEBUG_KMS("invalid link bandwidth in DPCD: %x (link count: %d)\n",
|
||||
dp_link_bw, dp_link_count);
|
||||
if (link_rate == 0 || link_lane_count == 0)
|
||||
DRM_DEBUG_KMS("invalid link rate/lane count: (%d / %d)\n",
|
||||
link_rate, link_lane_count);
|
||||
|
||||
return dp_link_bw * dp_link_count / 2;
|
||||
/* See DP v2.0 2.6.4.2, VCPayload_Bandwidth_for_OneTimeSlotPer_MTP_Allocation */
|
||||
return link_rate * link_lane_count / 54000;
|
||||
}
|
||||
EXPORT_SYMBOL(drm_dp_get_vc_payload_bw);
|
||||
|
||||
/**
|
||||
* drm_dp_read_mst_cap() - check whether or not a sink supports MST
|
||||
@ -3692,7 +3704,7 @@ int drm_dp_mst_topology_mgr_set_mst(struct drm_dp_mst_topology_mgr *mgr, bool ms
|
||||
goto out_unlock;
|
||||
}
|
||||
|
||||
mgr->pbn_div = drm_dp_get_vc_payload_bw(mgr->dpcd[1],
|
||||
mgr->pbn_div = drm_dp_get_vc_payload_bw(drm_dp_bw_code_to_link_rate(mgr->dpcd[1]),
|
||||
mgr->dpcd[2] & DP_MAX_LANE_COUNT_MASK);
|
||||
if (mgr->pbn_div == 0) {
|
||||
ret = -EINVAL;
|
||||
|
@ -276,6 +276,13 @@ int i9xx_check_plane_surface(struct intel_plane_state *plane_state)
|
||||
}
|
||||
}
|
||||
|
||||
if (IS_HASWELL(dev_priv) || IS_BROADWELL(dev_priv)) {
|
||||
drm_WARN_ON(&dev_priv->drm, src_x > 8191 || src_y > 4095);
|
||||
} else if (INTEL_GEN(dev_priv) >= 4 &&
|
||||
fb->modifier == I915_FORMAT_MOD_X_TILED) {
|
||||
drm_WARN_ON(&dev_priv->drm, src_x > 4095 || src_y > 4095);
|
||||
}
|
||||
|
||||
plane_state->color_plane[0].offset = offset;
|
||||
plane_state->color_plane[0].x = src_x;
|
||||
plane_state->color_plane[0].y = src_y;
|
||||
@ -488,6 +495,129 @@ static void i9xx_disable_plane(struct intel_plane *plane,
|
||||
spin_unlock_irqrestore(&dev_priv->uncore.lock, irqflags);
|
||||
}
|
||||
|
||||
static void
|
||||
g4x_primary_async_flip(struct intel_plane *plane,
|
||||
const struct intel_crtc_state *crtc_state,
|
||||
const struct intel_plane_state *plane_state,
|
||||
bool async_flip)
|
||||
{
|
||||
struct drm_i915_private *dev_priv = to_i915(plane->base.dev);
|
||||
u32 dspcntr = plane_state->ctl | i9xx_plane_ctl_crtc(crtc_state);
|
||||
u32 dspaddr_offset = plane_state->color_plane[0].offset;
|
||||
enum i9xx_plane_id i9xx_plane = plane->i9xx_plane;
|
||||
unsigned long irqflags;
|
||||
|
||||
if (async_flip)
|
||||
dspcntr |= DISPPLANE_ASYNC_FLIP;
|
||||
|
||||
spin_lock_irqsave(&dev_priv->uncore.lock, irqflags);
|
||||
intel_de_write_fw(dev_priv, DSPCNTR(i9xx_plane), dspcntr);
|
||||
intel_de_write_fw(dev_priv, DSPSURF(i9xx_plane),
|
||||
intel_plane_ggtt_offset(plane_state) + dspaddr_offset);
|
||||
spin_unlock_irqrestore(&dev_priv->uncore.lock, irqflags);
|
||||
}
|
||||
|
||||
static void
|
||||
vlv_primary_async_flip(struct intel_plane *plane,
|
||||
const struct intel_crtc_state *crtc_state,
|
||||
const struct intel_plane_state *plane_state,
|
||||
bool async_flip)
|
||||
{
|
||||
struct drm_i915_private *dev_priv = to_i915(plane->base.dev);
|
||||
u32 dspaddr_offset = plane_state->color_plane[0].offset;
|
||||
enum i9xx_plane_id i9xx_plane = plane->i9xx_plane;
|
||||
unsigned long irqflags;
|
||||
|
||||
spin_lock_irqsave(&dev_priv->uncore.lock, irqflags);
|
||||
intel_de_write_fw(dev_priv, DSPADDR_VLV(i9xx_plane),
|
||||
intel_plane_ggtt_offset(plane_state) + dspaddr_offset);
|
||||
spin_unlock_irqrestore(&dev_priv->uncore.lock, irqflags);
|
||||
}
|
||||
|
||||
static void
|
||||
bdw_primary_enable_flip_done(struct intel_plane *plane)
|
||||
{
|
||||
struct drm_i915_private *i915 = to_i915(plane->base.dev);
|
||||
enum pipe pipe = plane->pipe;
|
||||
|
||||
spin_lock_irq(&i915->irq_lock);
|
||||
bdw_enable_pipe_irq(i915, pipe, GEN8_PIPE_PRIMARY_FLIP_DONE);
|
||||
spin_unlock_irq(&i915->irq_lock);
|
||||
}
|
||||
|
||||
static void
|
||||
bdw_primary_disable_flip_done(struct intel_plane *plane)
|
||||
{
|
||||
struct drm_i915_private *i915 = to_i915(plane->base.dev);
|
||||
enum pipe pipe = plane->pipe;
|
||||
|
||||
spin_lock_irq(&i915->irq_lock);
|
||||
bdw_disable_pipe_irq(i915, pipe, GEN8_PIPE_PRIMARY_FLIP_DONE);
|
||||
spin_unlock_irq(&i915->irq_lock);
|
||||
}
|
||||
|
||||
static void
|
||||
ivb_primary_enable_flip_done(struct intel_plane *plane)
|
||||
{
|
||||
struct drm_i915_private *i915 = to_i915(plane->base.dev);
|
||||
|
||||
spin_lock_irq(&i915->irq_lock);
|
||||
ilk_enable_display_irq(i915, DE_PLANE_FLIP_DONE_IVB(plane->i9xx_plane));
|
||||
spin_unlock_irq(&i915->irq_lock);
|
||||
}
|
||||
|
||||
static void
|
||||
ivb_primary_disable_flip_done(struct intel_plane *plane)
|
||||
{
|
||||
struct drm_i915_private *i915 = to_i915(plane->base.dev);
|
||||
|
||||
spin_lock_irq(&i915->irq_lock);
|
||||
ilk_disable_display_irq(i915, DE_PLANE_FLIP_DONE_IVB(plane->i9xx_plane));
|
||||
spin_unlock_irq(&i915->irq_lock);
|
||||
}
|
||||
|
||||
static void
|
||||
ilk_primary_enable_flip_done(struct intel_plane *plane)
|
||||
{
|
||||
struct drm_i915_private *i915 = to_i915(plane->base.dev);
|
||||
|
||||
spin_lock_irq(&i915->irq_lock);
|
||||
ilk_enable_display_irq(i915, DE_PLANE_FLIP_DONE(plane->i9xx_plane));
|
||||
spin_unlock_irq(&i915->irq_lock);
|
||||
}
|
||||
|
||||
static void
|
||||
ilk_primary_disable_flip_done(struct intel_plane *plane)
|
||||
{
|
||||
struct drm_i915_private *i915 = to_i915(plane->base.dev);
|
||||
|
||||
spin_lock_irq(&i915->irq_lock);
|
||||
ilk_disable_display_irq(i915, DE_PLANE_FLIP_DONE(plane->i9xx_plane));
|
||||
spin_unlock_irq(&i915->irq_lock);
|
||||
}
|
||||
|
||||
static void
|
||||
vlv_primary_enable_flip_done(struct intel_plane *plane)
|
||||
{
|
||||
struct drm_i915_private *i915 = to_i915(plane->base.dev);
|
||||
enum pipe pipe = plane->pipe;
|
||||
|
||||
spin_lock_irq(&i915->irq_lock);
|
||||
i915_enable_pipestat(i915, pipe, PLANE_FLIP_DONE_INT_STATUS_VLV);
|
||||
spin_unlock_irq(&i915->irq_lock);
|
||||
}
|
||||
|
||||
static void
|
||||
vlv_primary_disable_flip_done(struct intel_plane *plane)
|
||||
{
|
||||
struct drm_i915_private *i915 = to_i915(plane->base.dev);
|
||||
enum pipe pipe = plane->pipe;
|
||||
|
||||
spin_lock_irq(&i915->irq_lock);
|
||||
i915_disable_pipestat(i915, pipe, PLANE_FLIP_DONE_INT_STATUS_VLV);
|
||||
spin_unlock_irq(&i915->irq_lock);
|
||||
}
|
||||
|
||||
static bool i9xx_plane_get_hw_state(struct intel_plane *plane,
|
||||
enum pipe *pipe)
|
||||
{
|
||||
@ -523,21 +653,56 @@ static bool i9xx_plane_get_hw_state(struct intel_plane *plane,
|
||||
return ret;
|
||||
}
|
||||
|
||||
static unsigned int
|
||||
hsw_primary_max_stride(struct intel_plane *plane,
|
||||
u32 pixel_format, u64 modifier,
|
||||
unsigned int rotation)
|
||||
{
|
||||
const struct drm_format_info *info = drm_format_info(pixel_format);
|
||||
int cpp = info->cpp[0];
|
||||
|
||||
/* Limit to 8k pixels to guarantee OFFSET.x doesn't get too big. */
|
||||
return min(8192 * cpp, 32 * 1024);
|
||||
}
|
||||
|
||||
static unsigned int
|
||||
ilk_primary_max_stride(struct intel_plane *plane,
|
||||
u32 pixel_format, u64 modifier,
|
||||
unsigned int rotation)
|
||||
{
|
||||
const struct drm_format_info *info = drm_format_info(pixel_format);
|
||||
int cpp = info->cpp[0];
|
||||
|
||||
/* Limit to 4k pixels to guarantee TILEOFF.x doesn't get too big. */
|
||||
if (modifier == I915_FORMAT_MOD_X_TILED)
|
||||
return min(4096 * cpp, 32 * 1024);
|
||||
else
|
||||
return 32 * 1024;
|
||||
}
|
||||
|
||||
unsigned int
|
||||
i965_plane_max_stride(struct intel_plane *plane,
|
||||
u32 pixel_format, u64 modifier,
|
||||
unsigned int rotation)
|
||||
{
|
||||
const struct drm_format_info *info = drm_format_info(pixel_format);
|
||||
int cpp = info->cpp[0];
|
||||
|
||||
/* Limit to 4k pixels to guarantee TILEOFF.x doesn't get too big. */
|
||||
if (modifier == I915_FORMAT_MOD_X_TILED)
|
||||
return min(4096 * cpp, 16 * 1024);
|
||||
else
|
||||
return 32 * 1024;
|
||||
}
|
||||
|
||||
static unsigned int
|
||||
i9xx_plane_max_stride(struct intel_plane *plane,
|
||||
u32 pixel_format, u64 modifier,
|
||||
unsigned int rotation)
|
||||
{
|
||||
struct drm_i915_private *dev_priv = to_i915(plane->base.dev);
|
||||
|
||||
if (!HAS_GMCH(dev_priv)) {
|
||||
return 32*1024;
|
||||
} else if (INTEL_GEN(dev_priv) >= 4) {
|
||||
if (modifier == I915_FORMAT_MOD_X_TILED)
|
||||
return 16*1024;
|
||||
else
|
||||
return 32*1024;
|
||||
} else if (INTEL_GEN(dev_priv) >= 3) {
|
||||
if (INTEL_GEN(dev_priv) >= 3) {
|
||||
if (modifier == I915_FORMAT_MOD_X_TILED)
|
||||
return 8*1024;
|
||||
else
|
||||
@ -649,12 +814,42 @@ intel_primary_plane_create(struct drm_i915_private *dev_priv, enum pipe pipe)
|
||||
else
|
||||
plane->min_cdclk = i9xx_plane_min_cdclk;
|
||||
|
||||
plane->max_stride = i9xx_plane_max_stride;
|
||||
if (HAS_GMCH(dev_priv)) {
|
||||
if (INTEL_GEN(dev_priv) >= 4)
|
||||
plane->max_stride = i965_plane_max_stride;
|
||||
else
|
||||
plane->max_stride = i9xx_plane_max_stride;
|
||||
} else {
|
||||
if (IS_BROADWELL(dev_priv) || IS_HASWELL(dev_priv))
|
||||
plane->max_stride = hsw_primary_max_stride;
|
||||
else
|
||||
plane->max_stride = ilk_primary_max_stride;
|
||||
}
|
||||
|
||||
plane->update_plane = i9xx_update_plane;
|
||||
plane->disable_plane = i9xx_disable_plane;
|
||||
plane->get_hw_state = i9xx_plane_get_hw_state;
|
||||
plane->check_plane = i9xx_plane_check;
|
||||
|
||||
if (IS_VALLEYVIEW(dev_priv) || IS_CHERRYVIEW(dev_priv)) {
|
||||
plane->async_flip = vlv_primary_async_flip;
|
||||
plane->enable_flip_done = vlv_primary_enable_flip_done;
|
||||
plane->disable_flip_done = vlv_primary_disable_flip_done;
|
||||
} else if (IS_BROADWELL(dev_priv)) {
|
||||
plane->need_async_flip_disable_wa = true;
|
||||
plane->async_flip = g4x_primary_async_flip;
|
||||
plane->enable_flip_done = bdw_primary_enable_flip_done;
|
||||
plane->disable_flip_done = bdw_primary_disable_flip_done;
|
||||
} else if (INTEL_GEN(dev_priv) >= 7) {
|
||||
plane->async_flip = g4x_primary_async_flip;
|
||||
plane->enable_flip_done = ivb_primary_enable_flip_done;
|
||||
plane->disable_flip_done = ivb_primary_disable_flip_done;
|
||||
} else if (INTEL_GEN(dev_priv) >= 5) {
|
||||
plane->async_flip = g4x_primary_async_flip;
|
||||
plane->enable_flip_done = ilk_primary_enable_flip_done;
|
||||
plane->disable_flip_done = ilk_primary_disable_flip_done;
|
||||
}
|
||||
|
||||
if (INTEL_GEN(dev_priv) >= 5 || IS_G4X(dev_priv))
|
||||
ret = drm_universal_plane_init(&dev_priv->drm, &plane->base,
|
||||
0, plane_funcs,
|
||||
|
@ -13,7 +13,7 @@ struct drm_i915_private;
|
||||
struct intel_plane;
|
||||
struct intel_plane_state;
|
||||
|
||||
unsigned int i9xx_plane_max_stride(struct intel_plane *plane,
|
||||
unsigned int i965_plane_max_stride(struct intel_plane *plane,
|
||||
u32 pixel_format, u64 modifier,
|
||||
unsigned int rotation);
|
||||
int i9xx_check_plane_surface(struct intel_plane_state *plane_state);
|
||||
|
@ -20,76 +20,9 @@ struct intel_qgv_point {
|
||||
struct intel_qgv_info {
|
||||
struct intel_qgv_point points[I915_NUM_QGV_POINTS];
|
||||
u8 num_points;
|
||||
u8 num_channels;
|
||||
u8 t_bl;
|
||||
enum intel_dram_type dram_type;
|
||||
};
|
||||
|
||||
static int icl_pcode_read_mem_global_info(struct drm_i915_private *dev_priv,
|
||||
struct intel_qgv_info *qi)
|
||||
{
|
||||
u32 val = 0;
|
||||
int ret;
|
||||
|
||||
ret = sandybridge_pcode_read(dev_priv,
|
||||
ICL_PCODE_MEM_SUBSYSYSTEM_INFO |
|
||||
ICL_PCODE_MEM_SS_READ_GLOBAL_INFO,
|
||||
&val, NULL);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (IS_GEN(dev_priv, 12)) {
|
||||
switch (val & 0xf) {
|
||||
case 0:
|
||||
qi->dram_type = INTEL_DRAM_DDR4;
|
||||
break;
|
||||
case 3:
|
||||
qi->dram_type = INTEL_DRAM_LPDDR4;
|
||||
break;
|
||||
case 4:
|
||||
qi->dram_type = INTEL_DRAM_DDR3;
|
||||
break;
|
||||
case 5:
|
||||
qi->dram_type = INTEL_DRAM_LPDDR3;
|
||||
break;
|
||||
default:
|
||||
MISSING_CASE(val & 0xf);
|
||||
break;
|
||||
}
|
||||
} else if (IS_GEN(dev_priv, 11)) {
|
||||
switch (val & 0xf) {
|
||||
case 0:
|
||||
qi->dram_type = INTEL_DRAM_DDR4;
|
||||
break;
|
||||
case 1:
|
||||
qi->dram_type = INTEL_DRAM_DDR3;
|
||||
break;
|
||||
case 2:
|
||||
qi->dram_type = INTEL_DRAM_LPDDR3;
|
||||
break;
|
||||
case 3:
|
||||
qi->dram_type = INTEL_DRAM_LPDDR4;
|
||||
break;
|
||||
default:
|
||||
MISSING_CASE(val & 0xf);
|
||||
break;
|
||||
}
|
||||
} else {
|
||||
MISSING_CASE(INTEL_GEN(dev_priv));
|
||||
qi->dram_type = INTEL_DRAM_LPDDR3; /* Conservative default */
|
||||
}
|
||||
|
||||
qi->num_channels = (val & 0xf0) >> 4;
|
||||
qi->num_points = (val & 0xf00) >> 8;
|
||||
|
||||
if (IS_GEN(dev_priv, 12))
|
||||
qi->t_bl = qi->dram_type == INTEL_DRAM_DDR4 ? 4 : 16;
|
||||
else if (IS_GEN(dev_priv, 11))
|
||||
qi->t_bl = qi->dram_type == INTEL_DRAM_DDR4 ? 4 : 8;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int icl_pcode_read_qgv_point_info(struct drm_i915_private *dev_priv,
|
||||
struct intel_qgv_point *sp,
|
||||
int point)
|
||||
@ -139,11 +72,15 @@ int icl_pcode_restrict_qgv_points(struct drm_i915_private *dev_priv,
|
||||
static int icl_get_qgv_points(struct drm_i915_private *dev_priv,
|
||||
struct intel_qgv_info *qi)
|
||||
{
|
||||
const struct dram_info *dram_info = &dev_priv->dram_info;
|
||||
int i, ret;
|
||||
|
||||
ret = icl_pcode_read_mem_global_info(dev_priv, qi);
|
||||
if (ret)
|
||||
return ret;
|
||||
qi->num_points = dram_info->num_qgv_points;
|
||||
|
||||
if (IS_GEN(dev_priv, 12))
|
||||
qi->t_bl = dev_priv->dram_info.type == INTEL_DRAM_DDR4 ? 4 : 16;
|
||||
else if (IS_GEN(dev_priv, 11))
|
||||
qi->t_bl = dev_priv->dram_info.type == INTEL_DRAM_DDR4 ? 4 : 8;
|
||||
|
||||
if (drm_WARN_ON(&dev_priv->drm,
|
||||
qi->num_points > ARRAY_SIZE(qi->points)))
|
||||
@ -209,7 +146,7 @@ static int icl_get_bw_info(struct drm_i915_private *dev_priv, const struct intel
|
||||
{
|
||||
struct intel_qgv_info qi = {};
|
||||
bool is_y_tile = true; /* assume y tile may be used */
|
||||
int num_channels;
|
||||
int num_channels = dev_priv->dram_info.num_channels;
|
||||
int deinterleave;
|
||||
int ipqdepth, ipqdepthpch;
|
||||
int dclk_max;
|
||||
@ -222,7 +159,6 @@ static int icl_get_bw_info(struct drm_i915_private *dev_priv, const struct intel
|
||||
"Failed to get memory subsystem information, ignoring bandwidth limits");
|
||||
return ret;
|
||||
}
|
||||
num_channels = qi.num_channels;
|
||||
|
||||
deinterleave = DIV_ROUND_UP(num_channels, is_y_tile ? 4 : 2);
|
||||
dclk_max = icl_sagv_max_dclk(&qi);
|
||||
|
@ -1317,6 +1317,11 @@ static unsigned int intel_linear_alignment(const struct drm_i915_private *dev_pr
|
||||
return 0;
|
||||
}
|
||||
|
||||
static bool has_async_flips(struct drm_i915_private *i915)
|
||||
{
|
||||
return INTEL_GEN(i915) >= 5;
|
||||
}
|
||||
|
||||
static unsigned int intel_surf_alignment(const struct drm_framebuffer *fb,
|
||||
int color_plane)
|
||||
{
|
||||
@ -1331,7 +1336,7 @@ static unsigned int intel_surf_alignment(const struct drm_framebuffer *fb,
|
||||
case DRM_FORMAT_MOD_LINEAR:
|
||||
return intel_linear_alignment(dev_priv);
|
||||
case I915_FORMAT_MOD_X_TILED:
|
||||
if (INTEL_GEN(dev_priv) >= 9)
|
||||
if (has_async_flips(dev_priv))
|
||||
return 256 * 1024;
|
||||
return 0;
|
||||
case I915_FORMAT_MOD_Y_TILED_GEN12_MC_CCS:
|
||||
@ -3086,6 +3091,8 @@ static int skl_check_main_surface(struct intel_plane_state *plane_state)
|
||||
}
|
||||
}
|
||||
|
||||
drm_WARN_ON(&dev_priv->drm, x > 8191 || y > 8191);
|
||||
|
||||
plane_state->color_plane[0].offset = offset;
|
||||
plane_state->color_plane[0].x = x;
|
||||
plane_state->color_plane[0].y = y;
|
||||
@ -3158,6 +3165,8 @@ static int skl_check_nv12_aux_surface(struct intel_plane_state *plane_state)
|
||||
}
|
||||
}
|
||||
|
||||
drm_WARN_ON(&i915->drm, x > 8191 || y > 8191);
|
||||
|
||||
plane_state->color_plane[uv_plane].offset = offset;
|
||||
plane_state->color_plane[uv_plane].x = x;
|
||||
plane_state->color_plane[uv_plane].y = y;
|
||||
@ -3577,6 +3586,8 @@ u32 glk_plane_color_ctl(const struct intel_crtc_state *crtc_state,
|
||||
plane_color_ctl |= PLANE_COLOR_YUV_RANGE_CORRECTION_DISABLE;
|
||||
} else if (fb->format->is_yuv) {
|
||||
plane_color_ctl |= PLANE_COLOR_INPUT_CSC_ENABLE;
|
||||
if (plane_state->hw.color_range == DRM_COLOR_YCBCR_FULL_RANGE)
|
||||
plane_color_ctl |= PLANE_COLOR_YUV_RANGE_CORRECTION_DISABLE;
|
||||
}
|
||||
|
||||
return plane_color_ctl;
|
||||
@ -14771,8 +14782,7 @@ static void intel_mode_config_init(struct drm_i915_private *i915)
|
||||
|
||||
mode_config->funcs = &intel_mode_funcs;
|
||||
|
||||
if (INTEL_GEN(i915) >= 9)
|
||||
mode_config->async_page_flip = true;
|
||||
mode_config->async_page_flip = has_async_flips(i915);
|
||||
|
||||
/*
|
||||
* Maximum framebuffer dimensions, chosen to match
|
||||
|
@ -722,16 +722,6 @@ static bool intel_dp_mst_get_qses_status(struct intel_digital_port *dig_port,
|
||||
return reply.auth_completed && reply.encryption_enabled;
|
||||
}
|
||||
|
||||
static
|
||||
bool intel_dp_mst_hdcp_check_link(struct intel_digital_port *dig_port,
|
||||
struct intel_connector *connector)
|
||||
{
|
||||
if (!intel_dp_hdcp_check_link(dig_port, connector))
|
||||
return false;
|
||||
|
||||
return intel_dp_mst_get_qses_status(dig_port, connector);
|
||||
}
|
||||
|
||||
static int
|
||||
intel_dp_mst_hdcp2_stream_encryption(struct intel_connector *connector,
|
||||
bool enable)
|
||||
@ -805,7 +795,7 @@ static const struct intel_hdcp_shim intel_dp_mst_hdcp_shim = {
|
||||
.read_v_prime_part = intel_dp_hdcp_read_v_prime_part,
|
||||
.toggle_signalling = intel_dp_hdcp_toggle_signalling,
|
||||
.stream_encryption = intel_dp_mst_hdcp_stream_encryption,
|
||||
.check_link = intel_dp_mst_hdcp_check_link,
|
||||
.check_link = intel_dp_hdcp_check_link,
|
||||
.hdcp_capable = intel_dp_hdcp_capable,
|
||||
.write_2_2_msg = intel_dp_hdcp2_write_msg,
|
||||
.read_2_2_msg = intel_dp_hdcp2_read_msg,
|
||||
|
@ -34,18 +34,6 @@ intel_dp_dump_link_status(const u8 link_status[DP_LINK_STATUS_SIZE])
|
||||
link_status[3], link_status[4], link_status[5]);
|
||||
}
|
||||
|
||||
static int intel_dp_lttpr_count(struct intel_dp *intel_dp)
|
||||
{
|
||||
int count = drm_dp_lttpr_count(intel_dp->lttpr_common_caps);
|
||||
|
||||
/*
|
||||
* Pretend no LTTPRs in case of LTTPR detection error, or
|
||||
* if too many (>8) LTTPRs are detected. This translates to link
|
||||
* training in transparent mode.
|
||||
*/
|
||||
return count <= 0 ? 0 : count;
|
||||
}
|
||||
|
||||
static void intel_dp_reset_lttpr_count(struct intel_dp *intel_dp)
|
||||
{
|
||||
intel_dp->lttpr_common_caps[DP_PHY_REPEATER_CNT -
|
||||
@ -142,6 +130,17 @@ int intel_dp_lttpr_init(struct intel_dp *intel_dp)
|
||||
return 0;
|
||||
|
||||
ret = intel_dp_read_lttpr_common_caps(intel_dp);
|
||||
if (!ret)
|
||||
return 0;
|
||||
|
||||
lttpr_count = drm_dp_lttpr_count(intel_dp->lttpr_common_caps);
|
||||
/*
|
||||
* Prevent setting LTTPR transparent mode explicitly if no LTTPRs are
|
||||
* detected as this breaks link training at least on the Dell WD19TB
|
||||
* dock.
|
||||
*/
|
||||
if (lttpr_count == 0)
|
||||
return 0;
|
||||
|
||||
/*
|
||||
* See DP Standard v2.0 3.6.6.1. about the explicit disabling of
|
||||
@ -150,17 +149,12 @@ int intel_dp_lttpr_init(struct intel_dp *intel_dp)
|
||||
*/
|
||||
intel_dp_set_lttpr_transparent_mode(intel_dp, true);
|
||||
|
||||
if (!ret)
|
||||
return 0;
|
||||
|
||||
lttpr_count = intel_dp_lttpr_count(intel_dp);
|
||||
|
||||
/*
|
||||
* In case of unsupported number of LTTPRs or failing to switch to
|
||||
* non-transparent mode fall-back to transparent link training mode,
|
||||
* still taking into account any LTTPR common lane- rate/count limits.
|
||||
*/
|
||||
if (lttpr_count == 0)
|
||||
if (lttpr_count < 0)
|
||||
return 0;
|
||||
|
||||
if (!intel_dp_set_lttpr_transparent_mode(intel_dp, false)) {
|
||||
@ -222,11 +216,11 @@ intel_dp_phy_is_downstream_of_source(struct intel_dp *intel_dp,
|
||||
enum drm_dp_phy dp_phy)
|
||||
{
|
||||
struct drm_i915_private *i915 = dp_to_i915(intel_dp);
|
||||
int lttpr_count = intel_dp_lttpr_count(intel_dp);
|
||||
int lttpr_count = drm_dp_lttpr_count(intel_dp->lttpr_common_caps);
|
||||
|
||||
drm_WARN_ON_ONCE(&i915->drm, lttpr_count == 0 && dp_phy != DP_PHY_DPRX);
|
||||
drm_WARN_ON_ONCE(&i915->drm, lttpr_count <= 0 && dp_phy != DP_PHY_DPRX);
|
||||
|
||||
return lttpr_count == 0 || dp_phy == DP_PHY_LTTPR(lttpr_count - 1);
|
||||
return lttpr_count <= 0 || dp_phy == DP_PHY_LTTPR(lttpr_count - 1);
|
||||
}
|
||||
|
||||
static u8 intel_dp_phy_voltage_max(struct intel_dp *intel_dp,
|
||||
|
@ -68,7 +68,9 @@ static int intel_dp_mst_compute_link_config(struct intel_encoder *encoder,
|
||||
|
||||
slots = drm_dp_atomic_find_vcpi_slots(state, &intel_dp->mst_mgr,
|
||||
connector->port,
|
||||
crtc_state->pbn, 0);
|
||||
crtc_state->pbn,
|
||||
drm_dp_get_vc_payload_bw(crtc_state->port_clock,
|
||||
crtc_state->lane_count));
|
||||
if (slots == -EDEADLK)
|
||||
return slots;
|
||||
if (slots >= 0)
|
||||
|
@ -649,13 +649,19 @@ skl_program_scaler(struct intel_plane *plane,
|
||||
|
||||
/* Preoffset values for YUV to RGB Conversion */
|
||||
#define PREOFF_YUV_TO_RGB_HI 0x1800
|
||||
#define PREOFF_YUV_TO_RGB_ME 0x1F00
|
||||
#define PREOFF_YUV_TO_RGB_ME 0x0000
|
||||
#define PREOFF_YUV_TO_RGB_LO 0x1800
|
||||
|
||||
#define ROFF(x) (((x) & 0xffff) << 16)
|
||||
#define GOFF(x) (((x) & 0xffff) << 0)
|
||||
#define BOFF(x) (((x) & 0xffff) << 16)
|
||||
|
||||
/*
|
||||
* Programs the input color space conversion stage for ICL HDR planes.
|
||||
* Note that it is assumed that this stage always happens after YUV
|
||||
* range correction. Thus, the input to this stage is assumed to be
|
||||
* in full-range YCbCr.
|
||||
*/
|
||||
static void
|
||||
icl_program_input_csc(struct intel_plane *plane,
|
||||
const struct intel_crtc_state *crtc_state,
|
||||
@ -703,52 +709,7 @@ icl_program_input_csc(struct intel_plane *plane,
|
||||
0x0, 0x7800, 0x7F10,
|
||||
},
|
||||
};
|
||||
|
||||
/* Matrix for Limited Range to Full Range Conversion */
|
||||
static const u16 input_csc_matrix_lr[][9] = {
|
||||
/*
|
||||
* BT.601 Limted range YCbCr -> full range RGB
|
||||
* The matrix required is :
|
||||
* [1.164384, 0.000, 1.596027,
|
||||
* 1.164384, -0.39175, -0.812813,
|
||||
* 1.164384, 2.017232, 0.0000]
|
||||
*/
|
||||
[DRM_COLOR_YCBCR_BT601] = {
|
||||
0x7CC8, 0x7950, 0x0,
|
||||
0x8D00, 0x7950, 0x9C88,
|
||||
0x0, 0x7950, 0x6810,
|
||||
},
|
||||
/*
|
||||
* BT.709 Limited range YCbCr -> full range RGB
|
||||
* The matrix required is :
|
||||
* [1.164384, 0.000, 1.792741,
|
||||
* 1.164384, -0.213249, -0.532909,
|
||||
* 1.164384, 2.112402, 0.0000]
|
||||
*/
|
||||
[DRM_COLOR_YCBCR_BT709] = {
|
||||
0x7E58, 0x7950, 0x0,
|
||||
0x8888, 0x7950, 0xADA8,
|
||||
0x0, 0x7950, 0x6870,
|
||||
},
|
||||
/*
|
||||
* BT.2020 Limited range YCbCr -> full range RGB
|
||||
* The matrix required is :
|
||||
* [1.164, 0.000, 1.678,
|
||||
* 1.164, -0.1873, -0.6504,
|
||||
* 1.164, 2.1417, 0.0000]
|
||||
*/
|
||||
[DRM_COLOR_YCBCR_BT2020] = {
|
||||
0x7D70, 0x7950, 0x0,
|
||||
0x8A68, 0x7950, 0xAC00,
|
||||
0x0, 0x7950, 0x6890,
|
||||
},
|
||||
};
|
||||
const u16 *csc;
|
||||
|
||||
if (plane_state->hw.color_range == DRM_COLOR_YCBCR_FULL_RANGE)
|
||||
csc = input_csc_matrix[plane_state->hw.color_encoding];
|
||||
else
|
||||
csc = input_csc_matrix_lr[plane_state->hw.color_encoding];
|
||||
const u16 *csc = input_csc_matrix[plane_state->hw.color_encoding];
|
||||
|
||||
intel_de_write_fw(dev_priv, PLANE_INPUT_CSC_COEFF(pipe, plane_id, 0),
|
||||
ROFF(csc[0]) | GOFF(csc[1]));
|
||||
@ -765,14 +726,8 @@ icl_program_input_csc(struct intel_plane *plane,
|
||||
|
||||
intel_de_write_fw(dev_priv, PLANE_INPUT_CSC_PREOFF(pipe, plane_id, 0),
|
||||
PREOFF_YUV_TO_RGB_HI);
|
||||
if (plane_state->hw.color_range == DRM_COLOR_YCBCR_FULL_RANGE)
|
||||
intel_de_write_fw(dev_priv,
|
||||
PLANE_INPUT_CSC_PREOFF(pipe, plane_id, 1),
|
||||
0);
|
||||
else
|
||||
intel_de_write_fw(dev_priv,
|
||||
PLANE_INPUT_CSC_PREOFF(pipe, plane_id, 1),
|
||||
PREOFF_YUV_TO_RGB_ME);
|
||||
intel_de_write_fw(dev_priv, PLANE_INPUT_CSC_PREOFF(pipe, plane_id, 1),
|
||||
PREOFF_YUV_TO_RGB_ME);
|
||||
intel_de_write_fw(dev_priv, PLANE_INPUT_CSC_PREOFF(pipe, plane_id, 2),
|
||||
PREOFF_YUV_TO_RGB_LO);
|
||||
intel_de_write_fw(dev_priv,
|
||||
@ -1896,7 +1851,26 @@ g4x_sprite_max_stride(struct intel_plane *plane,
|
||||
u32 pixel_format, u64 modifier,
|
||||
unsigned int rotation)
|
||||
{
|
||||
return 16384;
|
||||
const struct drm_format_info *info = drm_format_info(pixel_format);
|
||||
int cpp = info->cpp[0];
|
||||
|
||||
/* Limit to 4k pixels to guarantee TILEOFF.x doesn't get too big. */
|
||||
if (modifier == I915_FORMAT_MOD_X_TILED)
|
||||
return min(4096 * cpp, 16 * 1024);
|
||||
else
|
||||
return 16 * 1024;
|
||||
}
|
||||
|
||||
static unsigned int
|
||||
hsw_sprite_max_stride(struct intel_plane *plane,
|
||||
u32 pixel_format, u64 modifier,
|
||||
unsigned int rotation)
|
||||
{
|
||||
const struct drm_format_info *info = drm_format_info(pixel_format);
|
||||
int cpp = info->cpp[0];
|
||||
|
||||
/* Limit to 8k pixels to guarantee OFFSET.x doesn't get too big. */
|
||||
return min(8192 * cpp, 16 * 1024);
|
||||
}
|
||||
|
||||
static u32 g4x_sprite_ctl_crtc(const struct intel_crtc_state *crtc_state)
|
||||
@ -3453,11 +3427,11 @@ intel_sprite_plane_create(struct drm_i915_private *dev_priv,
|
||||
return plane;
|
||||
|
||||
if (IS_VALLEYVIEW(dev_priv) || IS_CHERRYVIEW(dev_priv)) {
|
||||
plane->max_stride = i9xx_plane_max_stride;
|
||||
plane->update_plane = vlv_update_plane;
|
||||
plane->disable_plane = vlv_disable_plane;
|
||||
plane->get_hw_state = vlv_plane_get_hw_state;
|
||||
plane->check_plane = vlv_sprite_check;
|
||||
plane->max_stride = i965_plane_max_stride;
|
||||
plane->min_cdclk = vlv_plane_min_cdclk;
|
||||
|
||||
if (IS_CHERRYVIEW(dev_priv) && pipe == PIPE_B) {
|
||||
@ -3471,16 +3445,18 @@ intel_sprite_plane_create(struct drm_i915_private *dev_priv,
|
||||
|
||||
plane_funcs = &vlv_sprite_funcs;
|
||||
} else if (INTEL_GEN(dev_priv) >= 7) {
|
||||
plane->max_stride = g4x_sprite_max_stride;
|
||||
plane->update_plane = ivb_update_plane;
|
||||
plane->disable_plane = ivb_disable_plane;
|
||||
plane->get_hw_state = ivb_plane_get_hw_state;
|
||||
plane->check_plane = g4x_sprite_check;
|
||||
|
||||
if (IS_BROADWELL(dev_priv) || IS_HASWELL(dev_priv))
|
||||
if (IS_BROADWELL(dev_priv) || IS_HASWELL(dev_priv)) {
|
||||
plane->max_stride = hsw_sprite_max_stride;
|
||||
plane->min_cdclk = hsw_plane_min_cdclk;
|
||||
else
|
||||
} else {
|
||||
plane->max_stride = g4x_sprite_max_stride;
|
||||
plane->min_cdclk = ivb_sprite_min_cdclk;
|
||||
}
|
||||
|
||||
formats = snb_plane_formats;
|
||||
num_formats = ARRAY_SIZE(snb_plane_formats);
|
||||
@ -3488,11 +3464,11 @@ intel_sprite_plane_create(struct drm_i915_private *dev_priv,
|
||||
|
||||
plane_funcs = &snb_sprite_funcs;
|
||||
} else {
|
||||
plane->max_stride = g4x_sprite_max_stride;
|
||||
plane->update_plane = g4x_update_plane;
|
||||
plane->disable_plane = g4x_disable_plane;
|
||||
plane->get_hw_state = g4x_plane_get_hw_state;
|
||||
plane->check_plane = g4x_sprite_check;
|
||||
plane->max_stride = g4x_sprite_max_stride;
|
||||
plane->min_cdclk = g4x_sprite_min_cdclk;
|
||||
|
||||
modifiers = i9xx_plane_format_modifiers;
|
||||
|
@ -609,14 +609,15 @@ static int i915_driver_hw_probe(struct drm_i915_private *dev_priv)
|
||||
goto err_msi;
|
||||
|
||||
intel_opregion_setup(dev_priv);
|
||||
/*
|
||||
* Fill the dram structure to get the system raw bandwidth and
|
||||
* dram info. This will be used for memory latency calculation.
|
||||
*/
|
||||
intel_dram_detect(dev_priv);
|
||||
|
||||
intel_pcode_init(dev_priv);
|
||||
|
||||
/*
|
||||
* Fill the dram structure to get the system dram info. This will be
|
||||
* used for memory latency calculation.
|
||||
*/
|
||||
intel_dram_detect(dev_priv);
|
||||
|
||||
intel_bw_init_hw(dev_priv);
|
||||
|
||||
return 0;
|
||||
|
@ -1125,11 +1125,8 @@ struct drm_i915_private {
|
||||
} wm;
|
||||
|
||||
struct dram_info {
|
||||
bool valid;
|
||||
bool is_16gb_dimm;
|
||||
bool wm_lv_0_adjust_needed;
|
||||
u8 num_channels;
|
||||
u8 ranks;
|
||||
u32 bandwidth_kbps;
|
||||
bool symmetric_memory;
|
||||
enum intel_dram_type {
|
||||
INTEL_DRAM_UNKNOWN,
|
||||
@ -1138,6 +1135,7 @@ struct drm_i915_private {
|
||||
INTEL_DRAM_LPDDR3,
|
||||
INTEL_DRAM_LPDDR4
|
||||
} type;
|
||||
u8 num_qgv_points;
|
||||
} dram_info;
|
||||
|
||||
struct intel_bw_info {
|
||||
|
@ -1540,6 +1540,9 @@ static void valleyview_pipestat_irq_handler(struct drm_i915_private *dev_priv,
|
||||
if (pipe_stats[pipe] & PIPE_START_VBLANK_INTERRUPT_STATUS)
|
||||
intel_handle_vblank(dev_priv, pipe);
|
||||
|
||||
if (pipe_stats[pipe] & PLANE_FLIP_DONE_INT_STATUS_VLV)
|
||||
flip_done_handler(dev_priv, pipe);
|
||||
|
||||
if (pipe_stats[pipe] & PIPE_CRC_DONE_INTERRUPT_STATUS)
|
||||
i9xx_pipe_crc_irq_handler(dev_priv, pipe);
|
||||
|
||||
@ -2052,6 +2055,9 @@ static void ilk_display_irq_handler(struct drm_i915_private *dev_priv,
|
||||
if (de_iir & DE_PIPE_VBLANK(pipe))
|
||||
intel_handle_vblank(dev_priv, pipe);
|
||||
|
||||
if (de_iir & DE_PLANE_FLIP_DONE(pipe))
|
||||
flip_done_handler(dev_priv, pipe);
|
||||
|
||||
if (de_iir & DE_PIPE_FIFO_UNDERRUN(pipe))
|
||||
intel_cpu_fifo_underrun_irq_handler(dev_priv, pipe);
|
||||
|
||||
@ -2104,6 +2110,9 @@ static void ivb_display_irq_handler(struct drm_i915_private *dev_priv,
|
||||
for_each_pipe(dev_priv, pipe) {
|
||||
if (de_iir & DE_PIPE_VBLANK_IVB(pipe))
|
||||
intel_handle_vblank(dev_priv, pipe);
|
||||
|
||||
if (de_iir & DE_PLANE_FLIP_DONE_IVB(pipe))
|
||||
flip_done_handler(dev_priv, pipe);
|
||||
}
|
||||
|
||||
/* check event from PCH */
|
||||
@ -2380,6 +2389,14 @@ static void gen11_dsi_te_interrupt_handler(struct drm_i915_private *dev_priv,
|
||||
intel_uncore_write(&dev_priv->uncore, DSI_INTR_IDENT_REG(port), tmp);
|
||||
}
|
||||
|
||||
static u32 gen8_de_pipe_flip_done_mask(struct drm_i915_private *i915)
|
||||
{
|
||||
if (INTEL_GEN(i915) >= 9)
|
||||
return GEN9_PIPE_PLANE1_FLIP_DONE;
|
||||
else
|
||||
return GEN8_PIPE_PRIMARY_FLIP_DONE;
|
||||
}
|
||||
|
||||
static irqreturn_t
|
||||
gen8_de_irq_handler(struct drm_i915_private *dev_priv, u32 master_ctl)
|
||||
{
|
||||
@ -2482,7 +2499,7 @@ gen8_de_irq_handler(struct drm_i915_private *dev_priv, u32 master_ctl)
|
||||
if (iir & GEN8_PIPE_VBLANK)
|
||||
intel_handle_vblank(dev_priv, pipe);
|
||||
|
||||
if (iir & GEN9_PIPE_PLANE1_FLIP_DONE)
|
||||
if (iir & gen8_de_pipe_flip_done_mask(dev_priv))
|
||||
flip_done_handler(dev_priv, pipe);
|
||||
|
||||
if (iir & GEN8_PIPE_CDCLK_CRC_DONE)
|
||||
@ -3101,13 +3118,10 @@ void gen8_irq_power_well_post_enable(struct drm_i915_private *dev_priv,
|
||||
u8 pipe_mask)
|
||||
{
|
||||
struct intel_uncore *uncore = &dev_priv->uncore;
|
||||
|
||||
u32 extra_ier = GEN8_PIPE_VBLANK | GEN8_PIPE_FIFO_UNDERRUN;
|
||||
u32 extra_ier = GEN8_PIPE_VBLANK | GEN8_PIPE_FIFO_UNDERRUN |
|
||||
gen8_de_pipe_flip_done_mask(dev_priv);
|
||||
enum pipe pipe;
|
||||
|
||||
if (INTEL_GEN(dev_priv) >= 9)
|
||||
extra_ier |= GEN9_PIPE_PLANE1_FLIP_DONE;
|
||||
|
||||
spin_lock_irq(&dev_priv->irq_lock);
|
||||
|
||||
if (!intel_irqs_enabled(dev_priv)) {
|
||||
@ -3582,6 +3596,9 @@ static void ilk_irq_postinstall(struct drm_i915_private *dev_priv)
|
||||
DE_PCH_EVENT_IVB | DE_AUX_CHANNEL_A_IVB);
|
||||
extra_mask = (DE_PIPEC_VBLANK_IVB | DE_PIPEB_VBLANK_IVB |
|
||||
DE_PIPEA_VBLANK_IVB | DE_ERR_INT_IVB |
|
||||
DE_PLANE_FLIP_DONE_IVB(PLANE_C) |
|
||||
DE_PLANE_FLIP_DONE_IVB(PLANE_B) |
|
||||
DE_PLANE_FLIP_DONE_IVB(PLANE_A) |
|
||||
DE_DP_A_HOTPLUG_IVB);
|
||||
} else {
|
||||
display_mask = (DE_MASTER_IRQ_CONTROL | DE_GSE | DE_PCH_EVENT |
|
||||
@ -3589,6 +3606,8 @@ static void ilk_irq_postinstall(struct drm_i915_private *dev_priv)
|
||||
DE_PIPEA_CRC_DONE | DE_POISON);
|
||||
extra_mask = (DE_PIPEA_VBLANK | DE_PIPEB_VBLANK |
|
||||
DE_PIPEB_FIFO_UNDERRUN | DE_PIPEA_FIFO_UNDERRUN |
|
||||
DE_PLANE_FLIP_DONE(PLANE_A) |
|
||||
DE_PLANE_FLIP_DONE(PLANE_B) |
|
||||
DE_DP_A_HOTPLUG);
|
||||
}
|
||||
|
||||
@ -3679,11 +3698,9 @@ static void gen8_de_irq_postinstall(struct drm_i915_private *dev_priv)
|
||||
de_port_masked |= DSI0_TE | DSI1_TE;
|
||||
}
|
||||
|
||||
de_pipe_enables = de_pipe_masked | GEN8_PIPE_VBLANK |
|
||||
GEN8_PIPE_FIFO_UNDERRUN;
|
||||
|
||||
if (INTEL_GEN(dev_priv) >= 9)
|
||||
de_pipe_enables |= GEN9_PIPE_PLANE1_FLIP_DONE;
|
||||
de_pipe_enables = de_pipe_masked |
|
||||
GEN8_PIPE_VBLANK | GEN8_PIPE_FIFO_UNDERRUN |
|
||||
gen8_de_pipe_flip_done_mask(dev_priv);
|
||||
|
||||
de_port_enables = de_port_masked;
|
||||
if (IS_GEN9_LP(dev_priv))
|
||||
|
@ -6578,6 +6578,7 @@ enum {
|
||||
#define TGL_CURSOR_D_OFFSET 0x73080
|
||||
|
||||
/* Display A control */
|
||||
#define _DSPAADDR_VLV 0x7017C /* vlv/chv */
|
||||
#define _DSPACNTR 0x70180
|
||||
#define DISPLAY_PLANE_ENABLE (1 << 31)
|
||||
#define DISPLAY_PLANE_DISABLE 0
|
||||
@ -6614,6 +6615,7 @@ enum {
|
||||
#define DISPPLANE_ROTATE_180 (1 << 15)
|
||||
#define DISPPLANE_TRICKLE_FEED_DISABLE (1 << 14) /* Ironlake */
|
||||
#define DISPPLANE_TILED (1 << 10)
|
||||
#define DISPPLANE_ASYNC_FLIP (1 << 9) /* g4x+ */
|
||||
#define DISPPLANE_MIRROR (1 << 8) /* CHV pipe B */
|
||||
#define _DSPAADDR 0x70184
|
||||
#define _DSPASTRIDE 0x70188
|
||||
@ -6625,6 +6627,7 @@ enum {
|
||||
#define _DSPASURFLIVE 0x701AC
|
||||
#define _DSPAGAMC 0x701E0
|
||||
|
||||
#define DSPADDR_VLV(plane) _MMIO_PIPE2(plane, _DSPAADDR_VLV)
|
||||
#define DSPCNTR(plane) _MMIO_PIPE2(plane, _DSPACNTR)
|
||||
#define DSPADDR(plane) _MMIO_PIPE2(plane, _DSPAADDR)
|
||||
#define DSPSTRIDE(plane) _MMIO_PIPE2(plane, _DSPASTRIDE)
|
||||
|
@ -5,6 +5,7 @@
|
||||
|
||||
#include "i915_drv.h"
|
||||
#include "intel_dram.h"
|
||||
#include "intel_sideband.h"
|
||||
|
||||
struct dram_dimm_info {
|
||||
u16 size;
|
||||
@ -201,22 +202,12 @@ skl_dram_get_channels_info(struct drm_i915_private *i915)
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/*
|
||||
* If any of the channel is single rank channel, worst case output
|
||||
* will be same as if single rank memory, so consider single rank
|
||||
* memory.
|
||||
*/
|
||||
if (ch0.ranks == 1 || ch1.ranks == 1)
|
||||
dram_info->ranks = 1;
|
||||
else
|
||||
dram_info->ranks = max(ch0.ranks, ch1.ranks);
|
||||
|
||||
if (dram_info->ranks == 0) {
|
||||
if (ch0.ranks == 0 && ch1.ranks == 0) {
|
||||
drm_info(&i915->drm, "couldn't get memory rank information\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
dram_info->is_16gb_dimm = ch0.is_16gb_dimm || ch1.is_16gb_dimm;
|
||||
dram_info->wm_lv_0_adjust_needed = ch0.is_16gb_dimm || ch1.is_16gb_dimm;
|
||||
|
||||
dram_info->symmetric_memory = intel_is_dram_symmetric(&ch0, &ch1);
|
||||
|
||||
@ -269,16 +260,12 @@ skl_get_dram_info(struct drm_i915_private *i915)
|
||||
mem_freq_khz = DIV_ROUND_UP((val & SKL_REQ_DATA_MASK) *
|
||||
SKL_MEMORY_FREQ_MULTIPLIER_HZ, 1000);
|
||||
|
||||
dram_info->bandwidth_kbps = dram_info->num_channels *
|
||||
mem_freq_khz * 8;
|
||||
|
||||
if (dram_info->bandwidth_kbps == 0) {
|
||||
if (dram_info->num_channels * mem_freq_khz == 0) {
|
||||
drm_info(&i915->drm,
|
||||
"Couldn't get system memory bandwidth\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
dram_info->valid = true;
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -365,7 +352,7 @@ static int bxt_get_dram_info(struct drm_i915_private *i915)
|
||||
struct dram_info *dram_info = &i915->dram_info;
|
||||
u32 dram_channels;
|
||||
u32 mem_freq_khz, val;
|
||||
u8 num_active_channels;
|
||||
u8 num_active_channels, valid_ranks = 0;
|
||||
int i;
|
||||
|
||||
val = intel_uncore_read(&i915->uncore, BXT_P_CR_MC_BIOS_REQ_0_0_0);
|
||||
@ -375,10 +362,7 @@ static int bxt_get_dram_info(struct drm_i915_private *i915)
|
||||
dram_channels = val & BXT_DRAM_CHANNEL_ACTIVE_MASK;
|
||||
num_active_channels = hweight32(dram_channels);
|
||||
|
||||
/* Each active bit represents 4-byte channel */
|
||||
dram_info->bandwidth_kbps = (mem_freq_khz * num_active_channels * 4);
|
||||
|
||||
if (dram_info->bandwidth_kbps == 0) {
|
||||
if (mem_freq_khz * num_active_channels == 0) {
|
||||
drm_info(&i915->drm,
|
||||
"Couldn't get system memory bandwidth\n");
|
||||
return -EINVAL;
|
||||
@ -410,57 +394,125 @@ static int bxt_get_dram_info(struct drm_i915_private *i915)
|
||||
dimm.size, dimm.width, dimm.ranks,
|
||||
intel_dram_type_str(type));
|
||||
|
||||
/*
|
||||
* If any of the channel is single rank channel,
|
||||
* worst case output will be same as if single rank
|
||||
* memory, so consider single rank memory.
|
||||
*/
|
||||
if (dram_info->ranks == 0)
|
||||
dram_info->ranks = dimm.ranks;
|
||||
else if (dimm.ranks == 1)
|
||||
dram_info->ranks = 1;
|
||||
if (valid_ranks == 0)
|
||||
valid_ranks = dimm.ranks;
|
||||
|
||||
if (type != INTEL_DRAM_UNKNOWN)
|
||||
dram_info->type = type;
|
||||
}
|
||||
|
||||
if (dram_info->type == INTEL_DRAM_UNKNOWN || dram_info->ranks == 0) {
|
||||
if (dram_info->type == INTEL_DRAM_UNKNOWN || valid_ranks == 0) {
|
||||
drm_info(&i915->drm, "couldn't get memory information\n");
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
dram_info->valid = true;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int icl_pcode_read_mem_global_info(struct drm_i915_private *dev_priv)
|
||||
{
|
||||
struct dram_info *dram_info = &dev_priv->dram_info;
|
||||
u32 val = 0;
|
||||
int ret;
|
||||
|
||||
ret = sandybridge_pcode_read(dev_priv,
|
||||
ICL_PCODE_MEM_SUBSYSYSTEM_INFO |
|
||||
ICL_PCODE_MEM_SS_READ_GLOBAL_INFO,
|
||||
&val, NULL);
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
if (IS_GEN(dev_priv, 12)) {
|
||||
switch (val & 0xf) {
|
||||
case 0:
|
||||
dram_info->type = INTEL_DRAM_DDR4;
|
||||
break;
|
||||
case 3:
|
||||
dram_info->type = INTEL_DRAM_LPDDR4;
|
||||
break;
|
||||
case 4:
|
||||
dram_info->type = INTEL_DRAM_DDR3;
|
||||
break;
|
||||
case 5:
|
||||
dram_info->type = INTEL_DRAM_LPDDR3;
|
||||
break;
|
||||
default:
|
||||
MISSING_CASE(val & 0xf);
|
||||
return -1;
|
||||
}
|
||||
} else {
|
||||
switch (val & 0xf) {
|
||||
case 0:
|
||||
dram_info->type = INTEL_DRAM_DDR4;
|
||||
break;
|
||||
case 1:
|
||||
dram_info->type = INTEL_DRAM_DDR3;
|
||||
break;
|
||||
case 2:
|
||||
dram_info->type = INTEL_DRAM_LPDDR3;
|
||||
break;
|
||||
case 3:
|
||||
dram_info->type = INTEL_DRAM_LPDDR4;
|
||||
break;
|
||||
default:
|
||||
MISSING_CASE(val & 0xf);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
dram_info->num_channels = (val & 0xf0) >> 4;
|
||||
dram_info->num_qgv_points = (val & 0xf00) >> 8;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int gen11_get_dram_info(struct drm_i915_private *i915)
|
||||
{
|
||||
int ret = skl_get_dram_info(i915);
|
||||
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
return icl_pcode_read_mem_global_info(i915);
|
||||
}
|
||||
|
||||
static int gen12_get_dram_info(struct drm_i915_private *i915)
|
||||
{
|
||||
/* Always needed for GEN12+ */
|
||||
i915->dram_info.wm_lv_0_adjust_needed = true;
|
||||
|
||||
return icl_pcode_read_mem_global_info(i915);
|
||||
}
|
||||
|
||||
void intel_dram_detect(struct drm_i915_private *i915)
|
||||
{
|
||||
struct dram_info *dram_info = &i915->dram_info;
|
||||
int ret;
|
||||
|
||||
/*
|
||||
* Assume 16Gb DIMMs are present until proven otherwise.
|
||||
* This is only used for the level 0 watermark latency
|
||||
* w/a which does not apply to bxt/glk.
|
||||
* Assume level 0 watermark latency adjustment is needed until proven
|
||||
* otherwise, this w/a is not needed by bxt/glk.
|
||||
*/
|
||||
dram_info->is_16gb_dimm = !IS_GEN9_LP(i915);
|
||||
dram_info->wm_lv_0_adjust_needed = !IS_GEN9_LP(i915);
|
||||
|
||||
if (INTEL_GEN(i915) < 9 || !HAS_DISPLAY(i915))
|
||||
return;
|
||||
|
||||
if (IS_GEN9_LP(i915))
|
||||
if (INTEL_GEN(i915) >= 12)
|
||||
ret = gen12_get_dram_info(i915);
|
||||
else if (INTEL_GEN(i915) >= 11)
|
||||
ret = gen11_get_dram_info(i915);
|
||||
else if (IS_GEN9_LP(i915))
|
||||
ret = bxt_get_dram_info(i915);
|
||||
else
|
||||
ret = skl_get_dram_info(i915);
|
||||
if (ret)
|
||||
return;
|
||||
|
||||
drm_dbg_kms(&i915->drm, "DRAM bandwidth: %u kBps, channels: %u\n",
|
||||
dram_info->bandwidth_kbps, dram_info->num_channels);
|
||||
drm_dbg_kms(&i915->drm, "DRAM channels: %u\n", dram_info->num_channels);
|
||||
|
||||
drm_dbg_kms(&i915->drm, "DRAM ranks: %u, 16Gb DIMMs: %s\n",
|
||||
dram_info->ranks, yesno(dram_info->is_16gb_dimm));
|
||||
drm_dbg_kms(&i915->drm, "Watermark level 0 adjustment needed: %s\n",
|
||||
yesno(dram_info->wm_lv_0_adjust_needed));
|
||||
}
|
||||
|
||||
static u32 gen9_edram_size_mb(struct drm_i915_private *i915, u32 cap)
|
||||
|
@ -2930,7 +2930,7 @@ static void intel_read_wm_latency(struct drm_i915_private *dev_priv,
|
||||
* any underrun. If not able to get Dimm info assume 16GB dimm
|
||||
* to avoid any underrun.
|
||||
*/
|
||||
if (dev_priv->dram_info.is_16gb_dimm)
|
||||
if (dev_priv->dram_info.wm_lv_0_adjust_needed)
|
||||
wm[0] += 1;
|
||||
|
||||
} else if (IS_HASWELL(dev_priv) || IS_BROADWELL(dev_priv)) {
|
||||
|
@ -783,6 +783,7 @@ drm_dp_mst_detect_port(struct drm_connector *connector,
|
||||
|
||||
struct edid *drm_dp_mst_get_edid(struct drm_connector *connector, struct drm_dp_mst_topology_mgr *mgr, struct drm_dp_mst_port *port);
|
||||
|
||||
int drm_dp_get_vc_payload_bw(int link_rate, int link_lane_count);
|
||||
|
||||
int drm_dp_calc_pbn_mode(int clock, int bpp, bool dsc);
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user