On 08/11/16 13:34, Russell King - ARM Linux wrote: > On Tue, Nov 08, 2016 at 01:32:15PM +0000, Russell King - ARM Linux wrote: >> Unfortunately, my drm-tda998x-devel branch is slightly out of date with >> these patches it's the original set of 10 patches. I've not pushed >> these ones out to that branch yet, as I've three additional patches on >> top of these which aren't "ready" for pushing out. > > Here's the delta between the branch and what I just posted: Great, thanks. I committed that on top and my framebuffer console over DVI is still working. I'm dubious that it's thorough enough to have value (sadly I have neither the time nor the motivation to fight with HDMI and I2S audio), but feel free to consider that a tested-by if you wish. Robin. > > diff --git a/drivers/gpu/drm/i2c/tda998x_drv.c b/drivers/gpu/drm/i2c/tda998x_drv.c > index 1b262a89b7e1..3a5e5c466972 100644 > --- a/drivers/gpu/drm/i2c/tda998x_drv.c > +++ b/drivers/gpu/drm/i2c/tda998x_drv.c > @@ -43,12 +43,12 @@ struct tda998x_priv { > u16 rev; > u8 current_page; > bool is_on; > - bool is_hdmi_sink; > - bool is_hdmi_config; > + bool supports_infoframes; > + bool sink_has_audio; > u8 vip_cntrl_0; > u8 vip_cntrl_1; > u8 vip_cntrl_2; > - unsigned long tdms_clock; > + unsigned long tmds_clock; > struct tda998x_audio_params audio_params; > > struct platform_device *audio_pdev; > @@ -774,7 +774,7 @@ tda998x_configure_audio(struct tda998x_priv *priv, > * assume 100MHz requires larger divider. > */ > adiv = AUDIO_DIV_SERCLK_8; > - if (priv->tdms_clock > 100000) > + if (priv->tmds_clock > 100000) > adiv++; /* AUDIO_DIV_SERCLK_16 */ > > /* S/PDIF asks for a larger divider */ > @@ -869,8 +869,7 @@ static int tda998x_audio_hw_params(struct device *dev, void *data, > } > > mutex_lock(&priv->audio_mutex); > - /* We must not program the TDA998x for audio if the sink is DVI. */ > - if (priv->is_hdmi_config) > + if (priv->supports_infoframes && priv->sink_has_audio) > ret = tda998x_configure_audio(priv, &audio); > else > ret = 0; > @@ -962,6 +961,27 @@ static int tda998x_connector_dpms(struct drm_connector *connector, int mode) > return drm_helper_connector_dpms(connector, mode); > } > > +static int tda998x_connector_fill_modes(struct drm_connector *connector, > + uint32_t maxX, uint32_t maxY) > +{ > + struct tda998x_priv *priv = conn_to_tda998x_priv(connector); > + int ret; > + > + mutex_lock(&priv->audio_mutex); > + ret = drm_helper_probe_single_connector_modes(connector, maxX, maxY); > + > + if (connector->edid_blob_ptr) { > + struct edid *edid = (void *)connector->edid_blob_ptr->data; > + > + priv->sink_has_audio = drm_detect_monitor_audio(edid); > + } else { > + priv->sink_has_audio = false; > + } > + mutex_unlock(&priv->audio_mutex); > + > + return ret; > +} > + > static enum drm_connector_status > tda998x_connector_detect(struct drm_connector *connector, bool force) > { > @@ -980,7 +1000,7 @@ static void tda998x_connector_destroy(struct drm_connector *connector) > static const struct drm_connector_funcs tda998x_connector_funcs = { > .dpms = tda998x_connector_dpms, > .reset = drm_atomic_helper_connector_reset, > - .fill_modes = drm_helper_probe_single_connector_modes, > + .fill_modes = tda998x_connector_fill_modes, > .detect = tda998x_connector_detect, > .destroy = tda998x_connector_destroy, > .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state, > @@ -1072,11 +1092,7 @@ static int tda998x_connector_get_modes(struct drm_connector *connector) > > drm_mode_connector_update_edid_property(connector, edid); > n = drm_add_edid_modes(connector, edid); > - priv->is_hdmi_sink = drm_detect_hdmi_monitor(edid); > - > - mutex_lock(&priv->audio_mutex); > drm_edid_to_eld(connector, edid); > - mutex_unlock(&priv->audio_mutex); > > kfree(edid); > > @@ -1350,8 +1366,22 @@ tda998x_encoder_mode_set(struct drm_encoder *encoder, > /* must be last register set: */ > reg_write(priv, REG_TBG_CNTRL_0, 0); > > - /* Only setup the info frames if the sink is HDMI */ > - if (priv->is_hdmi_sink) { > + priv->tmds_clock = adjusted_mode->clock; > + > + /* CEA-861B section 6 says that: > + * CEA version 1 (CEA-861) has no support for infoframes. > + * CEA version 2 (CEA-861A) supports version 1 AVI infoframes, > + * and optional basic audio. > + * CEA version 3 (CEA-861B) supports version 1 and 2 AVI infoframes, > + * and optional digital audio, with audio infoframes. > + * > + * Since we only support generation of version 2 AVI infoframes, > + * ignore CEA version 2 and below (iow, behave as if we're a > + * CEA-861 source.) > + */ > + priv->supports_infoframes = priv->connector.display_info.cea_rev >= 3; > + > + if (priv->supports_infoframes) { > /* We need to turn HDMI HDCP stuff on to get audio through */ > reg &= ~TBG_CNTRL_1_DWIN_DIS; > reg_write(priv, REG_TBG_CNTRL_1, reg); > @@ -1360,13 +1390,11 @@ tda998x_encoder_mode_set(struct drm_encoder *encoder, > > tda998x_write_avi(priv, adjusted_mode); > > - priv->tdms_clock = adjusted_mode->clock; > - > - if (priv->audio_params.format != AFMT_UNUSED) > + if (priv->audio_params.format != AFMT_UNUSED && > + priv->sink_has_audio) > tda998x_configure_audio(priv, &priv->audio_params); > } > > - priv->is_hdmi_config = priv->is_hdmi_sink; > mutex_unlock(&priv->audio_mutex); > } > > > _______________________________________________ dri-devel mailing list dri-devel@xxxxxxxxxxxxxxxxxxxxx https://lists.freedesktop.org/mailman/listinfo/dri-devel