Merge drm/drm-next into drm-misc-next
We haven't done any backmerge for a while due to the merge window, and it starts to become an issue for komeda. Let's bring 5.4-rc1 in. Signed-off-by: Maxime Ripard <mripard@kernel.org>
This commit is contained in:
@@ -44,15 +44,16 @@
|
||||
|
||||
#include "i915_debugfs.h"
|
||||
#include "i915_drv.h"
|
||||
#include "i915_trace.h"
|
||||
#include "intel_atomic.h"
|
||||
#include "intel_audio.h"
|
||||
#include "intel_connector.h"
|
||||
#include "intel_ddi.h"
|
||||
#include "intel_display_types.h"
|
||||
#include "intel_dp.h"
|
||||
#include "intel_dp_link_training.h"
|
||||
#include "intel_dp_mst.h"
|
||||
#include "intel_dpio_phy.h"
|
||||
#include "intel_drv.h"
|
||||
#include "intel_fifo_underrun.h"
|
||||
#include "intel_hdcp.h"
|
||||
#include "intel_hdmi.h"
|
||||
@@ -62,6 +63,7 @@
|
||||
#include "intel_panel.h"
|
||||
#include "intel_psr.h"
|
||||
#include "intel_sideband.h"
|
||||
#include "intel_tc.h"
|
||||
#include "intel_vdsc.h"
|
||||
|
||||
#define DP_DPRX_ESI_LEN 14
|
||||
@@ -211,46 +213,13 @@ static int intel_dp_max_common_rate(struct intel_dp *intel_dp)
|
||||
return intel_dp->common_rates[intel_dp->num_common_rates - 1];
|
||||
}
|
||||
|
||||
static int intel_dp_get_fia_supported_lane_count(struct intel_dp *intel_dp)
|
||||
{
|
||||
struct intel_digital_port *dig_port = dp_to_dig_port(intel_dp);
|
||||
struct drm_i915_private *dev_priv = to_i915(dig_port->base.base.dev);
|
||||
enum tc_port tc_port = intel_port_to_tc(dev_priv, dig_port->base.port);
|
||||
intel_wakeref_t wakeref;
|
||||
u32 lane_info;
|
||||
|
||||
if (tc_port == PORT_TC_NONE || dig_port->tc_type != TC_PORT_TYPEC)
|
||||
return 4;
|
||||
|
||||
lane_info = 0;
|
||||
with_intel_display_power(dev_priv, POWER_DOMAIN_DISPLAY_CORE, wakeref)
|
||||
lane_info = (I915_READ(PORT_TX_DFLEXDPSP) &
|
||||
DP_LANE_ASSIGNMENT_MASK(tc_port)) >>
|
||||
DP_LANE_ASSIGNMENT_SHIFT(tc_port);
|
||||
|
||||
switch (lane_info) {
|
||||
default:
|
||||
MISSING_CASE(lane_info);
|
||||
case 1:
|
||||
case 2:
|
||||
case 4:
|
||||
case 8:
|
||||
return 1;
|
||||
case 3:
|
||||
case 12:
|
||||
return 2;
|
||||
case 15:
|
||||
return 4;
|
||||
}
|
||||
}
|
||||
|
||||
/* Theoretical max between source and sink */
|
||||
static int intel_dp_max_common_lane_count(struct intel_dp *intel_dp)
|
||||
{
|
||||
struct intel_digital_port *intel_dig_port = dp_to_dig_port(intel_dp);
|
||||
int source_max = intel_dig_port->max_lanes;
|
||||
int sink_max = drm_dp_max_lane_count(intel_dp->dpcd);
|
||||
int fia_max = intel_dp_get_fia_supported_lane_count(intel_dp);
|
||||
int fia_max = intel_tc_port_fia_max_lane_count(intel_dig_port);
|
||||
|
||||
return min3(source_max, sink_max, fia_max);
|
||||
}
|
||||
@@ -329,9 +298,9 @@ static int icl_max_source_rate(struct intel_dp *intel_dp)
|
||||
{
|
||||
struct intel_digital_port *dig_port = dp_to_dig_port(intel_dp);
|
||||
struct drm_i915_private *dev_priv = to_i915(dig_port->base.base.dev);
|
||||
enum port port = dig_port->base.port;
|
||||
enum phy phy = intel_port_to_phy(dev_priv, dig_port->base.port);
|
||||
|
||||
if (intel_port_is_combophy(dev_priv, port) &&
|
||||
if (intel_phy_is_combo(dev_priv, phy) &&
|
||||
!IS_ELKHARTLAKE(dev_priv) &&
|
||||
!intel_dp_is_edp(intel_dp))
|
||||
return 540000;
|
||||
@@ -1208,7 +1177,7 @@ static u32 skl_get_aux_send_ctl(struct intel_dp *intel_dp,
|
||||
DP_AUX_CH_CTL_FW_SYNC_PULSE_SKL(32) |
|
||||
DP_AUX_CH_CTL_SYNC_PULSE_SKL(32);
|
||||
|
||||
if (intel_dig_port->tc_type == TC_PORT_TBT)
|
||||
if (intel_dig_port->tc_mode == TC_PORT_TBT_ALT)
|
||||
ret |= DP_AUX_CH_CTL_TBT_IO;
|
||||
|
||||
return ret;
|
||||
@@ -1224,6 +1193,8 @@ intel_dp_aux_xfer(struct intel_dp *intel_dp,
|
||||
struct drm_i915_private *i915 =
|
||||
to_i915(intel_dig_port->base.base.dev);
|
||||
struct intel_uncore *uncore = &i915->uncore;
|
||||
enum phy phy = intel_port_to_phy(i915, intel_dig_port->base.port);
|
||||
bool is_tc_port = intel_phy_is_tc(i915, phy);
|
||||
i915_reg_t ch_ctl, ch_data[5];
|
||||
u32 aux_clock_divider;
|
||||
enum intel_display_power_domain aux_domain =
|
||||
@@ -1239,6 +1210,9 @@ intel_dp_aux_xfer(struct intel_dp *intel_dp,
|
||||
for (i = 0; i < ARRAY_SIZE(ch_data); i++)
|
||||
ch_data[i] = intel_dp->aux_ch_data_reg(intel_dp, i);
|
||||
|
||||
if (is_tc_port)
|
||||
intel_tc_port_lock(intel_dig_port);
|
||||
|
||||
aux_wakeref = intel_display_power_get(i915, aux_domain);
|
||||
pps_wakeref = pps_lock(intel_dp);
|
||||
|
||||
@@ -1391,6 +1365,9 @@ out:
|
||||
pps_unlock(intel_dp, pps_wakeref);
|
||||
intel_display_power_put_async(i915, aux_domain, aux_wakeref);
|
||||
|
||||
if (is_tc_port)
|
||||
intel_tc_port_unlock(intel_dig_port);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
@@ -1878,8 +1855,10 @@ intel_dp_compute_link_config_wide(struct intel_dp *intel_dp,
|
||||
int mode_rate, link_clock, link_avail;
|
||||
|
||||
for (bpp = limits->max_bpp; bpp >= limits->min_bpp; bpp -= 2 * 3) {
|
||||
int output_bpp = intel_dp_output_bpp(pipe_config, bpp);
|
||||
|
||||
mode_rate = intel_dp_link_required(adjusted_mode->crtc_clock,
|
||||
bpp);
|
||||
output_bpp);
|
||||
|
||||
for (clock = limits->min_clock; clock <= limits->max_clock; clock++) {
|
||||
for (lane_count = limits->min_lane_count;
|
||||
@@ -2392,9 +2371,8 @@ static void wait_panel_status(struct intel_dp *intel_dp,
|
||||
I915_READ(pp_stat_reg),
|
||||
I915_READ(pp_ctrl_reg));
|
||||
|
||||
if (intel_wait_for_register(&dev_priv->uncore,
|
||||
pp_stat_reg, mask, value,
|
||||
5000))
|
||||
if (intel_de_wait_for_register(dev_priv, pp_stat_reg,
|
||||
mask, value, 5000))
|
||||
DRM_ERROR("Panel status timeout: status %08x control %08x\n",
|
||||
I915_READ(pp_stat_reg),
|
||||
I915_READ(pp_ctrl_reg));
|
||||
@@ -3981,10 +3959,8 @@ void intel_dp_set_idle_link_train(struct intel_dp *intel_dp)
|
||||
if (port == PORT_A)
|
||||
return;
|
||||
|
||||
if (intel_wait_for_register(&dev_priv->uncore, DP_TP_STATUS(port),
|
||||
DP_TP_STATUS_IDLE_DONE,
|
||||
DP_TP_STATUS_IDLE_DONE,
|
||||
1))
|
||||
if (intel_de_wait_for_set(dev_priv, DP_TP_STATUS(port),
|
||||
DP_TP_STATUS_IDLE_DONE, 1))
|
||||
DRM_ERROR("Timed out waiting for DP idle patterns\n");
|
||||
}
|
||||
|
||||
@@ -4168,10 +4144,6 @@ intel_edp_init_dpcd(struct intel_dp *intel_dp)
|
||||
drm_dp_read_desc(&intel_dp->aux, &intel_dp->desc,
|
||||
drm_dp_is_branch(intel_dp->dpcd));
|
||||
|
||||
if (intel_dp->dpcd[DP_DPCD_REV] >= 0x11)
|
||||
dev_priv->no_aux_handshake = intel_dp->dpcd[DP_MAX_DOWNSPREAD] &
|
||||
DP_NO_AUX_HANDSHAKE_LINK_TRAINING;
|
||||
|
||||
/*
|
||||
* Read the eDP display control registers.
|
||||
*
|
||||
@@ -4243,8 +4215,14 @@ intel_dp_get_dpcd(struct intel_dp *intel_dp)
|
||||
if (!intel_dp_read_dpcd(intel_dp))
|
||||
return false;
|
||||
|
||||
/* Don't clobber cached eDP rates. */
|
||||
/*
|
||||
* Don't clobber cached eDP rates. Also skip re-reading
|
||||
* the OUI/ID since we know it won't change.
|
||||
*/
|
||||
if (!intel_dp_is_edp(intel_dp)) {
|
||||
drm_dp_read_desc(&intel_dp->aux, &intel_dp->desc,
|
||||
drm_dp_is_branch(intel_dp->dpcd));
|
||||
|
||||
intel_dp_set_sink_rates(intel_dp);
|
||||
intel_dp_set_common_rates(intel_dp);
|
||||
}
|
||||
@@ -4253,7 +4231,8 @@ intel_dp_get_dpcd(struct intel_dp *intel_dp)
|
||||
* Some eDP panels do not set a valid value for sink count, that is why
|
||||
* it don't care about read it here and in intel_edp_init_dpcd().
|
||||
*/
|
||||
if (!intel_dp_is_edp(intel_dp)) {
|
||||
if (!intel_dp_is_edp(intel_dp) &&
|
||||
!drm_dp_has_quirk(&intel_dp->desc, DP_DPCD_QUIRK_NO_SINK_COUNT)) {
|
||||
u8 count;
|
||||
ssize_t r;
|
||||
|
||||
@@ -4878,14 +4857,16 @@ int intel_dp_retrain_link(struct intel_encoder *encoder,
|
||||
* retrain the link to get a picture. That's in case no
|
||||
* userspace component reacted to intermittent HPD dip.
|
||||
*/
|
||||
static bool intel_dp_hotplug(struct intel_encoder *encoder,
|
||||
struct intel_connector *connector)
|
||||
static enum intel_hotplug_state
|
||||
intel_dp_hotplug(struct intel_encoder *encoder,
|
||||
struct intel_connector *connector,
|
||||
bool irq_received)
|
||||
{
|
||||
struct drm_modeset_acquire_ctx ctx;
|
||||
bool changed;
|
||||
enum intel_hotplug_state state;
|
||||
int ret;
|
||||
|
||||
changed = intel_encoder_hotplug(encoder, connector);
|
||||
state = intel_encoder_hotplug(encoder, connector, irq_received);
|
||||
|
||||
drm_modeset_acquire_init(&ctx, 0);
|
||||
|
||||
@@ -4904,7 +4885,14 @@ static bool intel_dp_hotplug(struct intel_encoder *encoder,
|
||||
drm_modeset_acquire_fini(&ctx);
|
||||
WARN(ret, "Acquiring modeset locks failed with %i\n", ret);
|
||||
|
||||
return changed;
|
||||
/*
|
||||
* Keeping it consistent with intel_ddi_hotplug() and
|
||||
* intel_hdmi_hotplug().
|
||||
*/
|
||||
if (state == INTEL_HOTPLUG_UNCHANGED && irq_received)
|
||||
state = INTEL_HOTPLUG_RETRY;
|
||||
|
||||
return state;
|
||||
}
|
||||
|
||||
static void intel_dp_check_service_irq(struct intel_dp *intel_dp)
|
||||
@@ -5232,204 +5220,16 @@ static bool icl_combo_port_connected(struct drm_i915_private *dev_priv,
|
||||
return I915_READ(SDEISR) & SDE_DDI_HOTPLUG_ICP(port);
|
||||
}
|
||||
|
||||
static const char *tc_type_name(enum tc_port_type type)
|
||||
{
|
||||
static const char * const names[] = {
|
||||
[TC_PORT_UNKNOWN] = "unknown",
|
||||
[TC_PORT_LEGACY] = "legacy",
|
||||
[TC_PORT_TYPEC] = "typec",
|
||||
[TC_PORT_TBT] = "tbt",
|
||||
};
|
||||
|
||||
if (WARN_ON(type >= ARRAY_SIZE(names)))
|
||||
type = TC_PORT_UNKNOWN;
|
||||
|
||||
return names[type];
|
||||
}
|
||||
|
||||
static void icl_update_tc_port_type(struct drm_i915_private *dev_priv,
|
||||
struct intel_digital_port *intel_dig_port,
|
||||
bool is_legacy, bool is_typec, bool is_tbt)
|
||||
{
|
||||
enum port port = intel_dig_port->base.port;
|
||||
enum tc_port_type old_type = intel_dig_port->tc_type;
|
||||
|
||||
WARN_ON(is_legacy + is_typec + is_tbt != 1);
|
||||
|
||||
if (is_legacy)
|
||||
intel_dig_port->tc_type = TC_PORT_LEGACY;
|
||||
else if (is_typec)
|
||||
intel_dig_port->tc_type = TC_PORT_TYPEC;
|
||||
else if (is_tbt)
|
||||
intel_dig_port->tc_type = TC_PORT_TBT;
|
||||
else
|
||||
return;
|
||||
|
||||
/* Types are not supposed to be changed at runtime. */
|
||||
WARN_ON(old_type != TC_PORT_UNKNOWN &&
|
||||
old_type != intel_dig_port->tc_type);
|
||||
|
||||
if (old_type != intel_dig_port->tc_type)
|
||||
DRM_DEBUG_KMS("Port %c has TC type %s\n", port_name(port),
|
||||
tc_type_name(intel_dig_port->tc_type));
|
||||
}
|
||||
|
||||
/*
|
||||
* This function implements the first part of the Connect Flow described by our
|
||||
* specification, Gen11 TypeC Programming chapter. The rest of the flow (reading
|
||||
* lanes, EDID, etc) is done as needed in the typical places.
|
||||
*
|
||||
* Unlike the other ports, type-C ports are not available to use as soon as we
|
||||
* get a hotplug. The type-C PHYs can be shared between multiple controllers:
|
||||
* display, USB, etc. As a result, handshaking through FIA is required around
|
||||
* connect and disconnect to cleanly transfer ownership with the controller and
|
||||
* set the type-C power state.
|
||||
*
|
||||
* We could opt to only do the connect flow when we actually try to use the AUX
|
||||
* channels or do a modeset, then immediately run the disconnect flow after
|
||||
* usage, but there are some implications on this for a dynamic environment:
|
||||
* things may go away or change behind our backs. So for now our driver is
|
||||
* always trying to acquire ownership of the controller as soon as it gets an
|
||||
* interrupt (or polls state and sees a port is connected) and only gives it
|
||||
* back when it sees a disconnect. Implementation of a more fine-grained model
|
||||
* will require a lot of coordination with user space and thorough testing for
|
||||
* the extra possible cases.
|
||||
*/
|
||||
static bool icl_tc_phy_connect(struct drm_i915_private *dev_priv,
|
||||
struct intel_digital_port *dig_port)
|
||||
{
|
||||
enum tc_port tc_port = intel_port_to_tc(dev_priv, dig_port->base.port);
|
||||
u32 val;
|
||||
|
||||
if (dig_port->tc_type != TC_PORT_LEGACY &&
|
||||
dig_port->tc_type != TC_PORT_TYPEC)
|
||||
return true;
|
||||
|
||||
val = I915_READ(PORT_TX_DFLEXDPPMS);
|
||||
if (!(val & DP_PHY_MODE_STATUS_COMPLETED(tc_port))) {
|
||||
DRM_DEBUG_KMS("DP PHY for TC port %d not ready\n", tc_port);
|
||||
WARN_ON(dig_port->tc_legacy_port);
|
||||
return false;
|
||||
}
|
||||
|
||||
/*
|
||||
* This function may be called many times in a row without an HPD event
|
||||
* in between, so try to avoid the write when we can.
|
||||
*/
|
||||
val = I915_READ(PORT_TX_DFLEXDPCSSS);
|
||||
if (!(val & DP_PHY_MODE_STATUS_NOT_SAFE(tc_port))) {
|
||||
val |= DP_PHY_MODE_STATUS_NOT_SAFE(tc_port);
|
||||
I915_WRITE(PORT_TX_DFLEXDPCSSS, val);
|
||||
}
|
||||
|
||||
/*
|
||||
* Now we have to re-check the live state, in case the port recently
|
||||
* became disconnected. Not necessary for legacy mode.
|
||||
*/
|
||||
if (dig_port->tc_type == TC_PORT_TYPEC &&
|
||||
!(I915_READ(PORT_TX_DFLEXDPSP) & TC_LIVE_STATE_TC(tc_port))) {
|
||||
DRM_DEBUG_KMS("TC PHY %d sudden disconnect.\n", tc_port);
|
||||
icl_tc_phy_disconnect(dev_priv, dig_port);
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/*
|
||||
* See the comment at the connect function. This implements the Disconnect
|
||||
* Flow.
|
||||
*/
|
||||
void icl_tc_phy_disconnect(struct drm_i915_private *dev_priv,
|
||||
struct intel_digital_port *dig_port)
|
||||
{
|
||||
enum tc_port tc_port = intel_port_to_tc(dev_priv, dig_port->base.port);
|
||||
|
||||
if (dig_port->tc_type == TC_PORT_UNKNOWN)
|
||||
return;
|
||||
|
||||
/*
|
||||
* TBT disconnection flow is read the live status, what was done in
|
||||
* caller.
|
||||
*/
|
||||
if (dig_port->tc_type == TC_PORT_TYPEC ||
|
||||
dig_port->tc_type == TC_PORT_LEGACY) {
|
||||
u32 val;
|
||||
|
||||
val = I915_READ(PORT_TX_DFLEXDPCSSS);
|
||||
val &= ~DP_PHY_MODE_STATUS_NOT_SAFE(tc_port);
|
||||
I915_WRITE(PORT_TX_DFLEXDPCSSS, val);
|
||||
}
|
||||
|
||||
DRM_DEBUG_KMS("Port %c TC type %s disconnected\n",
|
||||
port_name(dig_port->base.port),
|
||||
tc_type_name(dig_port->tc_type));
|
||||
|
||||
dig_port->tc_type = TC_PORT_UNKNOWN;
|
||||
}
|
||||
|
||||
/*
|
||||
* The type-C ports are different because even when they are connected, they may
|
||||
* not be available/usable by the graphics driver: see the comment on
|
||||
* icl_tc_phy_connect(). So in our driver instead of adding the additional
|
||||
* concept of "usable" and make everything check for "connected and usable" we
|
||||
* define a port as "connected" when it is not only connected, but also when it
|
||||
* is usable by the rest of the driver. That maintains the old assumption that
|
||||
* connected ports are usable, and avoids exposing to the users objects they
|
||||
* can't really use.
|
||||
*/
|
||||
static bool icl_tc_port_connected(struct drm_i915_private *dev_priv,
|
||||
struct intel_digital_port *intel_dig_port)
|
||||
{
|
||||
enum port port = intel_dig_port->base.port;
|
||||
enum tc_port tc_port = intel_port_to_tc(dev_priv, port);
|
||||
bool is_legacy, is_typec, is_tbt;
|
||||
u32 dpsp;
|
||||
|
||||
/*
|
||||
* Complain if we got a legacy port HPD, but VBT didn't mark the port as
|
||||
* legacy. Treat the port as legacy from now on.
|
||||
*/
|
||||
if (!intel_dig_port->tc_legacy_port &&
|
||||
I915_READ(SDEISR) & SDE_TC_HOTPLUG_ICP(tc_port)) {
|
||||
DRM_ERROR("VBT incorrectly claims port %c is not TypeC legacy\n",
|
||||
port_name(port));
|
||||
intel_dig_port->tc_legacy_port = true;
|
||||
}
|
||||
is_legacy = intel_dig_port->tc_legacy_port;
|
||||
|
||||
/*
|
||||
* The spec says we shouldn't be using the ISR bits for detecting
|
||||
* between TC and TBT. We should use DFLEXDPSP.
|
||||
*/
|
||||
dpsp = I915_READ(PORT_TX_DFLEXDPSP);
|
||||
is_typec = dpsp & TC_LIVE_STATE_TC(tc_port);
|
||||
is_tbt = dpsp & TC_LIVE_STATE_TBT(tc_port);
|
||||
|
||||
if (!is_legacy && !is_typec && !is_tbt) {
|
||||
icl_tc_phy_disconnect(dev_priv, intel_dig_port);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
icl_update_tc_port_type(dev_priv, intel_dig_port, is_legacy, is_typec,
|
||||
is_tbt);
|
||||
|
||||
if (!icl_tc_phy_connect(dev_priv, intel_dig_port))
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static bool icl_digital_port_connected(struct intel_encoder *encoder)
|
||||
{
|
||||
struct drm_i915_private *dev_priv = to_i915(encoder->base.dev);
|
||||
struct intel_digital_port *dig_port = enc_to_dig_port(&encoder->base);
|
||||
enum phy phy = intel_port_to_phy(dev_priv, encoder->port);
|
||||
|
||||
if (intel_port_is_combophy(dev_priv, encoder->port))
|
||||
if (intel_phy_is_combo(dev_priv, phy))
|
||||
return icl_combo_port_connected(dev_priv, dig_port);
|
||||
else if (intel_port_is_tc(dev_priv, encoder->port))
|
||||
return icl_tc_port_connected(dev_priv, dig_port);
|
||||
else if (intel_phy_is_tc(dev_priv, phy))
|
||||
return intel_tc_port_connected(dig_port);
|
||||
else
|
||||
MISSING_CASE(encoder->hpd_pin);
|
||||
|
||||
@@ -5587,9 +5387,6 @@ intel_dp_detect(struct drm_connector *connector,
|
||||
if (INTEL_GEN(dev_priv) >= 11)
|
||||
intel_dp_get_dsc_sink_cap(intel_dp);
|
||||
|
||||
drm_dp_read_desc(&intel_dp->aux, &intel_dp->desc,
|
||||
drm_dp_is_branch(intel_dp->dpcd));
|
||||
|
||||
intel_dp_configure_mst(intel_dp);
|
||||
|
||||
if (intel_dp->is_mst) {
|
||||
@@ -6013,47 +5810,49 @@ struct hdcp2_dp_errata_stream_type {
|
||||
u8 stream_type;
|
||||
} __packed;
|
||||
|
||||
static struct hdcp2_dp_msg_data {
|
||||
struct hdcp2_dp_msg_data {
|
||||
u8 msg_id;
|
||||
u32 offset;
|
||||
bool msg_detectable;
|
||||
u32 timeout;
|
||||
u32 timeout2; /* Added for non_paired situation */
|
||||
} hdcp2_msg_data[] = {
|
||||
{HDCP_2_2_AKE_INIT, DP_HDCP_2_2_AKE_INIT_OFFSET, false, 0, 0},
|
||||
{HDCP_2_2_AKE_SEND_CERT, DP_HDCP_2_2_AKE_SEND_CERT_OFFSET,
|
||||
false, HDCP_2_2_CERT_TIMEOUT_MS, 0},
|
||||
{HDCP_2_2_AKE_NO_STORED_KM, DP_HDCP_2_2_AKE_NO_STORED_KM_OFFSET,
|
||||
false, 0, 0},
|
||||
{HDCP_2_2_AKE_STORED_KM, DP_HDCP_2_2_AKE_STORED_KM_OFFSET,
|
||||
false, 0, 0},
|
||||
{HDCP_2_2_AKE_SEND_HPRIME, DP_HDCP_2_2_AKE_SEND_HPRIME_OFFSET,
|
||||
true, HDCP_2_2_HPRIME_PAIRED_TIMEOUT_MS,
|
||||
HDCP_2_2_HPRIME_NO_PAIRED_TIMEOUT_MS},
|
||||
{HDCP_2_2_AKE_SEND_PAIRING_INFO,
|
||||
DP_HDCP_2_2_AKE_SEND_PAIRING_INFO_OFFSET, true,
|
||||
HDCP_2_2_PAIRING_TIMEOUT_MS, 0},
|
||||
{HDCP_2_2_LC_INIT, DP_HDCP_2_2_LC_INIT_OFFSET, false, 0, 0},
|
||||
{HDCP_2_2_LC_SEND_LPRIME, DP_HDCP_2_2_LC_SEND_LPRIME_OFFSET,
|
||||
false, HDCP_2_2_DP_LPRIME_TIMEOUT_MS, 0},
|
||||
{HDCP_2_2_SKE_SEND_EKS, DP_HDCP_2_2_SKE_SEND_EKS_OFFSET, false,
|
||||
0, 0},
|
||||
{HDCP_2_2_REP_SEND_RECVID_LIST,
|
||||
DP_HDCP_2_2_REP_SEND_RECVID_LIST_OFFSET, true,
|
||||
HDCP_2_2_RECVID_LIST_TIMEOUT_MS, 0},
|
||||
{HDCP_2_2_REP_SEND_ACK, DP_HDCP_2_2_REP_SEND_ACK_OFFSET, false,
|
||||
0, 0},
|
||||
{HDCP_2_2_REP_STREAM_MANAGE,
|
||||
DP_HDCP_2_2_REP_STREAM_MANAGE_OFFSET, false,
|
||||
0, 0},
|
||||
{HDCP_2_2_REP_STREAM_READY, DP_HDCP_2_2_REP_STREAM_READY_OFFSET,
|
||||
false, HDCP_2_2_STREAM_READY_TIMEOUT_MS, 0},
|
||||
};
|
||||
|
||||
static const struct hdcp2_dp_msg_data hdcp2_dp_msg_data[] = {
|
||||
{ HDCP_2_2_AKE_INIT, DP_HDCP_2_2_AKE_INIT_OFFSET, false, 0, 0 },
|
||||
{ HDCP_2_2_AKE_SEND_CERT, DP_HDCP_2_2_AKE_SEND_CERT_OFFSET,
|
||||
false, HDCP_2_2_CERT_TIMEOUT_MS, 0 },
|
||||
{ HDCP_2_2_AKE_NO_STORED_KM, DP_HDCP_2_2_AKE_NO_STORED_KM_OFFSET,
|
||||
false, 0, 0 },
|
||||
{ HDCP_2_2_AKE_STORED_KM, DP_HDCP_2_2_AKE_STORED_KM_OFFSET,
|
||||
false, 0, 0 },
|
||||
{ HDCP_2_2_AKE_SEND_HPRIME, DP_HDCP_2_2_AKE_SEND_HPRIME_OFFSET,
|
||||
true, HDCP_2_2_HPRIME_PAIRED_TIMEOUT_MS,
|
||||
HDCP_2_2_HPRIME_NO_PAIRED_TIMEOUT_MS },
|
||||
{ HDCP_2_2_AKE_SEND_PAIRING_INFO,
|
||||
DP_HDCP_2_2_AKE_SEND_PAIRING_INFO_OFFSET, true,
|
||||
HDCP_2_2_PAIRING_TIMEOUT_MS, 0 },
|
||||
{ HDCP_2_2_LC_INIT, DP_HDCP_2_2_LC_INIT_OFFSET, false, 0, 0 },
|
||||
{ HDCP_2_2_LC_SEND_LPRIME, DP_HDCP_2_2_LC_SEND_LPRIME_OFFSET,
|
||||
false, HDCP_2_2_DP_LPRIME_TIMEOUT_MS, 0 },
|
||||
{ HDCP_2_2_SKE_SEND_EKS, DP_HDCP_2_2_SKE_SEND_EKS_OFFSET, false,
|
||||
0, 0 },
|
||||
{ HDCP_2_2_REP_SEND_RECVID_LIST,
|
||||
DP_HDCP_2_2_REP_SEND_RECVID_LIST_OFFSET, true,
|
||||
HDCP_2_2_RECVID_LIST_TIMEOUT_MS, 0 },
|
||||
{ HDCP_2_2_REP_SEND_ACK, DP_HDCP_2_2_REP_SEND_ACK_OFFSET, false,
|
||||
0, 0 },
|
||||
{ HDCP_2_2_REP_STREAM_MANAGE,
|
||||
DP_HDCP_2_2_REP_STREAM_MANAGE_OFFSET, false,
|
||||
0, 0 },
|
||||
{ HDCP_2_2_REP_STREAM_READY, DP_HDCP_2_2_REP_STREAM_READY_OFFSET,
|
||||
false, HDCP_2_2_STREAM_READY_TIMEOUT_MS, 0 },
|
||||
/* local define to shovel this through the write_2_2 interface */
|
||||
#define HDCP_2_2_ERRATA_DP_STREAM_TYPE 50
|
||||
{HDCP_2_2_ERRATA_DP_STREAM_TYPE,
|
||||
DP_HDCP_2_2_REG_STREAM_TYPE_OFFSET, false,
|
||||
0, 0},
|
||||
};
|
||||
{ HDCP_2_2_ERRATA_DP_STREAM_TYPE,
|
||||
DP_HDCP_2_2_REG_STREAM_TYPE_OFFSET, false,
|
||||
0, 0 },
|
||||
};
|
||||
|
||||
static inline
|
||||
int intel_dp_hdcp2_read_rx_status(struct intel_digital_port *intel_dig_port,
|
||||
@@ -6107,7 +5906,7 @@ int hdcp2_detect_msg_availability(struct intel_digital_port *intel_dig_port,
|
||||
|
||||
static ssize_t
|
||||
intel_dp_hdcp2_wait_for_msg(struct intel_digital_port *intel_dig_port,
|
||||
struct hdcp2_dp_msg_data *hdcp2_msg_data)
|
||||
const struct hdcp2_dp_msg_data *hdcp2_msg_data)
|
||||
{
|
||||
struct intel_dp *dp = &intel_dig_port->dp;
|
||||
struct intel_hdcp *hdcp = &dp->attached_connector->hdcp;
|
||||
@@ -6146,13 +5945,13 @@ intel_dp_hdcp2_wait_for_msg(struct intel_digital_port *intel_dig_port,
|
||||
return ret;
|
||||
}
|
||||
|
||||
static struct hdcp2_dp_msg_data *get_hdcp2_dp_msg_data(u8 msg_id)
|
||||
static const struct hdcp2_dp_msg_data *get_hdcp2_dp_msg_data(u8 msg_id)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < ARRAY_SIZE(hdcp2_msg_data); i++)
|
||||
if (hdcp2_msg_data[i].msg_id == msg_id)
|
||||
return &hdcp2_msg_data[i];
|
||||
for (i = 0; i < ARRAY_SIZE(hdcp2_dp_msg_data); i++)
|
||||
if (hdcp2_dp_msg_data[i].msg_id == msg_id)
|
||||
return &hdcp2_dp_msg_data[i];
|
||||
|
||||
return NULL;
|
||||
}
|
||||
@@ -6166,7 +5965,7 @@ int intel_dp_hdcp2_write_msg(struct intel_digital_port *intel_dig_port,
|
||||
unsigned int offset;
|
||||
u8 *byte = buf;
|
||||
ssize_t ret, bytes_to_write, len;
|
||||
struct hdcp2_dp_msg_data *hdcp2_msg_data;
|
||||
const struct hdcp2_dp_msg_data *hdcp2_msg_data;
|
||||
|
||||
hdcp2_msg_data = get_hdcp2_dp_msg_data(*byte);
|
||||
if (!hdcp2_msg_data)
|
||||
@@ -6230,7 +6029,7 @@ int intel_dp_hdcp2_read_msg(struct intel_digital_port *intel_dig_port,
|
||||
unsigned int offset;
|
||||
u8 *byte = buf;
|
||||
ssize_t ret, bytes_to_recv, len;
|
||||
struct hdcp2_dp_msg_data *hdcp2_msg_data;
|
||||
const struct hdcp2_dp_msg_data *hdcp2_msg_data;
|
||||
|
||||
hdcp2_msg_data = get_hdcp2_dp_msg_data(msg_id);
|
||||
if (!hdcp2_msg_data)
|
||||
@@ -6832,8 +6631,6 @@ static void intel_dp_set_drrs_state(struct drm_i915_private *dev_priv,
|
||||
const struct intel_crtc_state *crtc_state,
|
||||
int refresh_rate)
|
||||
{
|
||||
struct intel_encoder *encoder;
|
||||
struct intel_digital_port *dig_port = NULL;
|
||||
struct intel_dp *intel_dp = dev_priv->drrs.dp;
|
||||
struct intel_crtc *intel_crtc = to_intel_crtc(crtc_state->base.crtc);
|
||||
enum drrs_refresh_rate_type index = DRRS_HIGH_RR;
|
||||
@@ -6848,9 +6645,6 @@ static void intel_dp_set_drrs_state(struct drm_i915_private *dev_priv,
|
||||
return;
|
||||
}
|
||||
|
||||
dig_port = dp_to_dig_port(intel_dp);
|
||||
encoder = &dig_port->base;
|
||||
|
||||
if (!intel_crtc) {
|
||||
DRM_DEBUG_KMS("DRRS: intel_crtc not initialized\n");
|
||||
return;
|
||||
@@ -7330,6 +7124,7 @@ intel_dp_init_connector(struct intel_digital_port *intel_dig_port,
|
||||
struct drm_device *dev = intel_encoder->base.dev;
|
||||
struct drm_i915_private *dev_priv = to_i915(dev);
|
||||
enum port port = intel_encoder->port;
|
||||
enum phy phy = intel_port_to_phy(dev_priv, port);
|
||||
int type;
|
||||
|
||||
/* Initialize the work for modeset in case of link train failure */
|
||||
@@ -7356,7 +7151,7 @@ intel_dp_init_connector(struct intel_digital_port *intel_dig_port,
|
||||
* Currently we don't support eDP on TypeC ports, although in
|
||||
* theory it could work on TypeC legacy ports.
|
||||
*/
|
||||
WARN_ON(intel_port_is_tc(dev_priv, port));
|
||||
WARN_ON(intel_phy_is_tc(dev_priv, phy));
|
||||
type = DRM_MODE_CONNECTOR_eDP;
|
||||
} else {
|
||||
type = DRM_MODE_CONNECTOR_DisplayPort;
|
||||
|
||||
Reference in New Issue
Block a user