[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <18be3d32-c643-42de-bb7d-165b409c8b24@quicinc.com>
Date: Fri, 7 Feb 2025 14:27:41 -0800
From: Abhinav Kumar <quic_abhinavk@...cinc.com>
To: Dmitry Baryshkov <dmitry.baryshkov@...aro.org>
CC: Andrzej Hajda <andrzej.hajda@...el.com>,
Neil Armstrong
<neil.armstrong@...aro.org>,
Robert Foss <rfoss@...nel.org>,
Laurent Pinchart
<Laurent.pinchart@...asonboard.com>,
Jonas Karlman <jonas@...boo.se>,
Jernej
Skrabec <jernej.skrabec@...il.com>,
Maarten Lankhorst
<maarten.lankhorst@...ux.intel.com>,
Maxime Ripard <mripard@...nel.org>,
Thomas Zimmermann <tzimmermann@...e.de>,
David Airlie <airlied@...il.com>, Rob Clark <robdclark@...il.com>,
Sean Paul <sean@...rly.run>,
Marijn Suijten
<marijn.suijten@...ainline.org>,
Simona Vetter <simona@...ll.ch>,
Simona
Vetter <simona.vetter@...ll.ch>,
<dri-devel@...ts.freedesktop.org>, <linux-arm-msm@...r.kernel.org>,
<freedreno@...ts.freedesktop.org>, <linux-kernel@...r.kernel.org>
Subject: Re: [PATCH v6 3/7] drm/msm/hdmi: make use of the drm_connector_hdmi
framework
On 2/7/2025 2:03 PM, Dmitry Baryshkov wrote:
> On Fri, Feb 07, 2025 at 01:34:35PM -0800, Abhinav Kumar wrote:
>>
>>
>> On 2/3/2025 5:30 PM, Dmitry Baryshkov wrote:
>>> On Mon, Feb 03, 2025 at 04:25:59PM -0800, Abhinav Kumar wrote:
>>>>
>>>>
>>>> On 1/24/2025 1:47 PM, Dmitry Baryshkov wrote:
>>>>> Setup the HDMI connector on the MSM HDMI outputs. Make use of
>>>>> atomic_check hook and of the provided Infoframe infrastructure.
>>>>>
>>>>
>>>> By atomic_check are you referring to the
>>>> msm_hdmi_bridge_tmds_char_rate_valid()?
>>>
>>> No, I mean drm_atomic_helper_connector_hdmi_check() being called from
>>> drm_bridge_connector (inthe previous versions it was called from this
>>> driver).
>>>
>>
>> Ack.
>>>>
>>>> Also please confirm if HDMI audio was re-tested with these changes.
>>>
>>> Yes, although not the channels allocation for the multi-channel audio. I
>>> don't have corresponding equipment. If you think that we should start
>>> testing that, I will check if I can get the 6.1 or 8.1 receiver and the
>>> speakers :-)
>>>
>>
>> We should but I am fine with basic audio validation for now.
>>
>>>>
>>>>> Acked-by: Maxime Ripard <mripard@...nel.org>
>>>>> Signed-off-by: Dmitry Baryshkov <dmitry.baryshkov@...aro.org>
>>>>> ---
>>>>> drivers/gpu/drm/msm/Kconfig | 2 +
>>>>> drivers/gpu/drm/msm/hdmi/hdmi.c | 45 ++-------
>>>>> drivers/gpu/drm/msm/hdmi/hdmi.h | 16 +--
>>>>> drivers/gpu/drm/msm/hdmi/hdmi_audio.c | 74 ++++----------
>>>>> drivers/gpu/drm/msm/hdmi/hdmi_bridge.c | 180 +++++++++++++++++++++++----------
>>>>> 5 files changed, 162 insertions(+), 155 deletions(-)
>>>>>
>>>>> diff --git a/drivers/gpu/drm/msm/Kconfig b/drivers/gpu/drm/msm/Kconfig
>>>>> index 7ec833b6d8292f8cb26cfe5582812f2754cd4d35..974bc7c0ea761147d3326bdce9039d6f26f290d0 100644
>>>>> --- a/drivers/gpu/drm/msm/Kconfig
>>>>> +++ b/drivers/gpu/drm/msm/Kconfig
>>>>> @@ -170,6 +170,8 @@ config DRM_MSM_HDMI
>>>>> bool "Enable HDMI support in MSM DRM driver"
>>>>> depends on DRM_MSM
>>>>> default y
>>>>> + select DRM_DISPLAY_HDMI_HELPER
>>>>> + select DRM_DISPLAY_HDMI_STATE_HELPER
>>>>> help
>>>>> Compile in support for the HDMI output MSM DRM driver. It can
>>>>> be a primary or a secondary display on device. Note that this is used
>>>>> diff --git a/drivers/gpu/drm/msm/hdmi/hdmi.c b/drivers/gpu/drm/msm/hdmi/hdmi.c
>>>>> index 37b3809c6bdd7c35aca6b475cb1f41c0ab4d3e6d..b14205cb9e977edd0d849e0eafe9b69c0da594bd 100644
>>>>> --- a/drivers/gpu/drm/msm/hdmi/hdmi.c
>>>>> +++ b/drivers/gpu/drm/msm/hdmi/hdmi.c
>>>>> @@ -12,6 +12,7 @@
>>>>> #include <drm/drm_bridge_connector.h>
>>>>> #include <drm/drm_of.h>
>>>>> +#include <drm/display/drm_hdmi_state_helper.h>
>>>>> #include <sound/hdmi-codec.h>
>>>>> #include "hdmi.h"
>>>>> @@ -165,8 +166,6 @@ int msm_hdmi_modeset_init(struct hdmi *hdmi,
>>>>> hdmi->dev = dev;
>>>>> hdmi->encoder = encoder;
>>>>> - hdmi_audio_infoframe_init(&hdmi->audio.infoframe);
>>>>> -
>>>>> ret = msm_hdmi_bridge_init(hdmi);
>>>>> if (ret) {
>>>>> DRM_DEV_ERROR(dev->dev, "failed to create HDMI bridge: %d\n", ret);
>>>>> @@ -254,40 +253,12 @@ static int msm_hdmi_audio_hw_params(struct device *dev, void *data,
>>>>> struct hdmi_codec_params *params)
>>>>> {
>>>>> struct hdmi *hdmi = dev_get_drvdata(dev);
>>>>> - unsigned int chan;
>>>>> - unsigned int channel_allocation = 0;
>>>>> unsigned int rate;
>>>>> - unsigned int level_shift = 0; /* 0dB */
>>>>> - bool down_mix = false;
>>>>> + int ret;
>>>>> DRM_DEV_DEBUG(dev, "%u Hz, %d bit, %d channels\n", params->sample_rate,
>>>>> params->sample_width, params->cea.channels);
>>>>> - switch (params->cea.channels) {
>>>>> - case 2:
>>>>> - /* FR and FL speakers */
>>>>> - channel_allocation = 0;
>>>>> - chan = MSM_HDMI_AUDIO_CHANNEL_2;
>>>>> - break;
>>>>> - case 4:
>>>>> - /* FC, LFE, FR and FL speakers */
>>>>> - channel_allocation = 0x3;
>>>>> - chan = MSM_HDMI_AUDIO_CHANNEL_4;
>>>>> - break;
>>>>> - case 6:
>>>>> - /* RR, RL, FC, LFE, FR and FL speakers */
>>>>> - channel_allocation = 0x0B;
>>>>> - chan = MSM_HDMI_AUDIO_CHANNEL_6;
>>>>> - break;
>>>>> - case 8:
>>>>> - /* FRC, FLC, RR, RL, FC, LFE, FR and FL speakers */
>>>>> - channel_allocation = 0x1F;
>>>>> - chan = MSM_HDMI_AUDIO_CHANNEL_8;
>>>>> - break;
>>>>> - default:
>>>>> - return -EINVAL;
>>>>> - }
>>>>
>>>> This mapping table was doing two things:
>>>>
>>>> 1) drop the conversion of channels to an index to nchannels[] and use the
>>>> channels directly. This part is fine but could have been a separate change
>>>> by itself to show the redundancy.
>>>>
>>>> 2) drop the mapping table of channels to channel_allocation.
>>>> I am not fully sure of this. Was this done because
>>>> hdmi_codec_channel_alloc[] table in hdmi-codec does this for us?
>>>> hdmi_codec_get_ch_alloc_table_idx() uses channels and the flags to come up
>>>> with the idx into this table. But it seems like current MSM HDMI code did
>>>> not consider the flags. So for example, it seems like for 6 channels, we
>>>> could return any of the below based on the flags but MSM HDMI always used
>>>> 0x0B so will the values match?
>>>
>>> Do they have to match? The correct value is being calculated by the HDMI
>>> code in ASoC and then being written into the Audio InfoFrame. If the MSM
>>> HDMI code wasn't taking ELD data into account, then it's a bug. But... I
>>> don't think it's worth spending too much time on fixing it separately.
>>>
>>
>> Its a change in values which are passed to the audio infoframe but I dont
>> know if it will change the behavior of what has been working so far (I hope
>> not).
>>
>> I agree, that msm hdmi driver not considering the flags in the mask is a bug
>> so for that reason, we should use the channels and channel allocation
>> supplied to us by hdmi_codec.
>>
>> If it does result in an issue (due to it incorrectly working today), we will
>> atleast know where to look at.
>>
>> Some more questions below.
>
> Ack
>
>>
>>> In the end, that is exactly the purpose of the frameworks - to make code
>>> error prone and to remove the need to reimplement same things over and
>>> over again, making differnt kinds of mistakes. For example, MSM HDMI
>>> code also doesn't implement plugged_cb support. It doesn't provide ELD
>>> to the HDMI codec code, etc. All of that is being fixed by using the
>>> framework. It's not worth implementing those functions in the MSM HDMI
>>> code first only to drop them in the next commit.
>>>
>>>>
>>>> 202 { .ca_id = 0x0b, .n_ch = 6,
>>>> 203 .mask = FL | FR | LFE | FC | RL | RR},
>>>> 204 /* surround40 */
>>>> 205 { .ca_id = 0x08, .n_ch = 6,
>>>> 206 .mask = FL | FR | RL | RR },
>>>> 207 /* surround41 */
>>>> 208 { .ca_id = 0x09, .n_ch = 6,
>>>> 209 .mask = FL | FR | LFE | RL | RR },
>>>> 210 /* surround50 */
>>>> 211 { .ca_id = 0x0a, .n_ch = 6,
>>>> 212 .mask = FL | FR | FC | RL | RR },
>>>> 213 /* 6.1 */
>>>>
>>>>
>>>>> -
>>>>> switch (params->sample_rate) {
>>>>> case 32000:
>>>>> rate = HDMI_SAMPLE_RATE_32KHZ;
>>>>> @@ -316,9 +287,12 @@ static int msm_hdmi_audio_hw_params(struct device *dev, void *data,
>>>>> return -EINVAL;
>>>>> }
>>>>> - msm_hdmi_audio_set_sample_rate(hdmi, rate);
>>>>> - msm_hdmi_audio_info_setup(hdmi, 1, chan, channel_allocation,
>>>>> - level_shift, down_mix);
>>>>> + ret = drm_atomic_helper_connector_hdmi_update_audio_infoframe(hdmi->connector,
>>>>> + ¶ms->cea);
>>>>> + if (ret)
>>>>> + return ret;
>>>>> +
>>>>> + msm_hdmi_audio_info_setup(hdmi, rate, params->cea.channels);
>>>>> return 0;
>>>>> }
>>>>> @@ -327,7 +301,8 @@ static void msm_hdmi_audio_shutdown(struct device *dev, void *data)
>>>>> {
>>>>> struct hdmi *hdmi = dev_get_drvdata(dev);
>>>>> - msm_hdmi_audio_info_setup(hdmi, 0, 0, 0, 0, 0);
>>>>> + drm_atomic_helper_connector_hdmi_clear_audio_infoframe(hdmi->connector);
>>>>> + msm_hdmi_audio_disable(hdmi);
>>>>> }
>>>>> static const struct hdmi_codec_ops msm_hdmi_audio_codec_ops = {
>>>>> diff --git a/drivers/gpu/drm/msm/hdmi/hdmi.h b/drivers/gpu/drm/msm/hdmi/hdmi.h
>>>>> index a62d2aedfbb7239d37c826c4f96762f100a2be4a..53b52351d0eddf4a5c87a5290016bb53ed4d29f7 100644
>>>>> --- a/drivers/gpu/drm/msm/hdmi/hdmi.h
>>>>> +++ b/drivers/gpu/drm/msm/hdmi/hdmi.h
>>>>> @@ -24,8 +24,8 @@ struct hdmi_platform_config;
>>>>> struct hdmi_audio {
>>>>> bool enabled;
>>>>> - struct hdmi_audio_infoframe infoframe;
>>>>> int rate;
>>>>> + int channels;
>>>>> };
>>>>> struct hdmi_hdcp_ctrl;
>>>>> @@ -207,12 +207,6 @@ static inline int msm_hdmi_pll_8998_init(struct platform_device *pdev)
>>>>> /*
>>>>> * audio:
>>>>> */
>>>>> -/* Supported HDMI Audio channels and rates */
>>>>> -#define MSM_HDMI_AUDIO_CHANNEL_2 0
>>>>> -#define MSM_HDMI_AUDIO_CHANNEL_4 1
>>>>> -#define MSM_HDMI_AUDIO_CHANNEL_6 2
>>>>> -#define MSM_HDMI_AUDIO_CHANNEL_8 3
>>>>> -
>>>>> #define HDMI_SAMPLE_RATE_32KHZ 0
>>>>> #define HDMI_SAMPLE_RATE_44_1KHZ 1
>>>>> #define HDMI_SAMPLE_RATE_48KHZ 2
>>>>> @@ -221,12 +215,8 @@ static inline int msm_hdmi_pll_8998_init(struct platform_device *pdev)
>>>>> #define HDMI_SAMPLE_RATE_176_4KHZ 5
>>>>> #define HDMI_SAMPLE_RATE_192KHZ 6
>>>>> -int msm_hdmi_audio_update(struct hdmi *hdmi);
>>>>> -int msm_hdmi_audio_info_setup(struct hdmi *hdmi, bool enabled,
>>>>> - uint32_t num_of_channels, uint32_t channel_allocation,
>>>>> - uint32_t level_shift, bool down_mix);
>>>>> -void msm_hdmi_audio_set_sample_rate(struct hdmi *hdmi, int rate);
>>>>> -
>>>>> +int msm_hdmi_audio_info_setup(struct hdmi *hdmi, int rate, int channels);
>>>>> +int msm_hdmi_audio_disable(struct hdmi *hdmi);
>>>>> /*
>>>>> * hdmi bridge:
>>>>> diff --git a/drivers/gpu/drm/msm/hdmi/hdmi_audio.c b/drivers/gpu/drm/msm/hdmi/hdmi_audio.c
>>>>> index 4c2058c4adc1001a12e10f35e88a6d58f3bd2fdc..924654bfb48cf17feadea1c0661ee6ee4e1b4589 100644
>>>>> --- a/drivers/gpu/drm/msm/hdmi/hdmi_audio.c
>>>>> +++ b/drivers/gpu/drm/msm/hdmi/hdmi_audio.c
>>>>> @@ -7,9 +7,6 @@
>>>>> #include <linux/hdmi.h>
>>>>> #include "hdmi.h"
>>>>> -/* maps MSM_HDMI_AUDIO_CHANNEL_n consts used by audio driver to # of channels: */
>>>>> -static int nchannels[] = { 2, 4, 6, 8 };
>>>>> -
>>>>> /* Supported HDMI Audio sample rates */
>>>>> #define MSM_HDMI_SAMPLE_RATE_32KHZ 0
>>>>> #define MSM_HDMI_SAMPLE_RATE_44_1KHZ 1
>>>>> @@ -71,19 +68,20 @@ static const struct hdmi_msm_audio_arcs *get_arcs(unsigned long int pixclock)
>>>>> return NULL;
>>>>> }
>>>>> -int msm_hdmi_audio_update(struct hdmi *hdmi)
>>>>> +static int msm_hdmi_audio_update(struct hdmi *hdmi)
>>>>> {
>>>>> struct hdmi_audio *audio = &hdmi->audio;
>>>>> - struct hdmi_audio_infoframe *info = &audio->infoframe;
>>>>> const struct hdmi_msm_audio_arcs *arcs = NULL;
>>>>> bool enabled = audio->enabled;
>>>>> uint32_t acr_pkt_ctrl, vbi_pkt_ctrl, aud_pkt_ctrl;
>>>>> - uint32_t infofrm_ctrl, audio_config;
>>>>> + uint32_t audio_config;
>>>>> +
>>>>> + if (!hdmi->connector->display_info.is_hdmi)
>>>>> + return -EINVAL;
>>>>> +
>>>>> + DBG("audio: enabled=%d, channels=%d, rate=%d",
>>>>> + audio->enabled, audio->channels, audio->rate);
>>>>> - DBG("audio: enabled=%d, channels=%d, channel_allocation=0x%x, "
>>>>> - "level_shift_value=%d, downmix_inhibit=%d, rate=%d",
>>>>> - audio->enabled, info->channels, info->channel_allocation,
>>>>> - info->level_shift_value, info->downmix_inhibit, audio->rate);
>>>>> DBG("video: power_on=%d, pixclock=%lu", hdmi->power_on, hdmi->pixclock);
>>>>> if (enabled && !(hdmi->power_on && hdmi->pixclock)) {
>>>>> @@ -104,7 +102,6 @@ int msm_hdmi_audio_update(struct hdmi *hdmi)
>>>>> acr_pkt_ctrl = hdmi_read(hdmi, REG_HDMI_ACR_PKT_CTRL);
>>>>> vbi_pkt_ctrl = hdmi_read(hdmi, REG_HDMI_VBI_PKT_CTRL);
>>>>> aud_pkt_ctrl = hdmi_read(hdmi, REG_HDMI_AUDIO_PKT_CTRL1);
>>>>> - infofrm_ctrl = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL0);
>>>>> audio_config = hdmi_read(hdmi, REG_HDMI_AUDIO_CFG);
>>>>> /* Clear N/CTS selection bits */
>>>>> @@ -113,7 +110,6 @@ int msm_hdmi_audio_update(struct hdmi *hdmi)
>>>>> if (enabled) {
>>>>> uint32_t n, cts, multiplier;
>>>>> enum hdmi_acr_cts select;
>>>>> - uint8_t buf[14];
>>>>> n = arcs->lut[audio->rate].n;
>>>>> cts = arcs->lut[audio->rate].cts;
>>>>> @@ -155,20 +151,12 @@ int msm_hdmi_audio_update(struct hdmi *hdmi)
>>>>> HDMI_ACR_1_N(n));
>>>>> hdmi_write(hdmi, REG_HDMI_AUDIO_PKT_CTRL2,
>>>>> - COND(info->channels != 2, HDMI_AUDIO_PKT_CTRL2_LAYOUT) |
>>>>> + COND(audio->channels != 2, HDMI_AUDIO_PKT_CTRL2_LAYOUT) |
>>>>> HDMI_AUDIO_PKT_CTRL2_OVERRIDE);
>>>>> acr_pkt_ctrl |= HDMI_ACR_PKT_CTRL_CONT;
>>>>> acr_pkt_ctrl |= HDMI_ACR_PKT_CTRL_SEND;
>>>>> - /* configure infoframe: */
>>>>> - hdmi_audio_infoframe_pack(info, buf, sizeof(buf));
>>>>> - hdmi_write(hdmi, REG_HDMI_AUDIO_INFO0,
>>>>> - (buf[3] << 0) | (buf[4] << 8) |
>>>>> - (buf[5] << 16) | (buf[6] << 24));
>>>>> - hdmi_write(hdmi, REG_HDMI_AUDIO_INFO1,
>>>>> - (buf[7] << 0) | (buf[8] << 8));
>>>>> -
>>>>> hdmi_write(hdmi, REG_HDMI_GC, 0);
>>>>> vbi_pkt_ctrl |= HDMI_VBI_PKT_CTRL_GC_ENABLE;
>>>>> @@ -176,11 +164,6 @@ int msm_hdmi_audio_update(struct hdmi *hdmi)
>>>>> aud_pkt_ctrl |= HDMI_AUDIO_PKT_CTRL1_AUDIO_SAMPLE_SEND;
>>>>> - infofrm_ctrl |= HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SEND;
>>>>> - infofrm_ctrl |= HDMI_INFOFRAME_CTRL0_AUDIO_INFO_CONT;
>>>>> - infofrm_ctrl |= HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SOURCE;
>>>>> - infofrm_ctrl |= HDMI_INFOFRAME_CTRL0_AUDIO_INFO_UPDATE;
>>>>> -
>>>>> audio_config &= ~HDMI_AUDIO_CFG_FIFO_WATERMARK__MASK;
>>>>> audio_config |= HDMI_AUDIO_CFG_FIFO_WATERMARK(4);
>>>>> audio_config |= HDMI_AUDIO_CFG_ENGINE_ENABLE;
>>>>> @@ -190,17 +173,12 @@ int msm_hdmi_audio_update(struct hdmi *hdmi)
>>>>> vbi_pkt_ctrl &= ~HDMI_VBI_PKT_CTRL_GC_ENABLE;
>>>>> vbi_pkt_ctrl &= ~HDMI_VBI_PKT_CTRL_GC_EVERY_FRAME;
>>>>> aud_pkt_ctrl &= ~HDMI_AUDIO_PKT_CTRL1_AUDIO_SAMPLE_SEND;
>>>>> - infofrm_ctrl &= ~HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SEND;
>>>>> - infofrm_ctrl &= ~HDMI_INFOFRAME_CTRL0_AUDIO_INFO_CONT;
>>>>> - infofrm_ctrl &= ~HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SOURCE;
>>>>> - infofrm_ctrl &= ~HDMI_INFOFRAME_CTRL0_AUDIO_INFO_UPDATE;
>>>>> audio_config &= ~HDMI_AUDIO_CFG_ENGINE_ENABLE;
>>>>> }
>>>>> hdmi_write(hdmi, REG_HDMI_ACR_PKT_CTRL, acr_pkt_ctrl);
>>>>> hdmi_write(hdmi, REG_HDMI_VBI_PKT_CTRL, vbi_pkt_ctrl);
>>>>> hdmi_write(hdmi, REG_HDMI_AUDIO_PKT_CTRL1, aud_pkt_ctrl);
>>>>> - hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL0, infofrm_ctrl);
>>>>> hdmi_write(hdmi, REG_HDMI_AUD_INT,
>>>>> COND(enabled, HDMI_AUD_INT_AUD_FIFO_URUN_INT) |
>>>>> @@ -214,41 +192,29 @@ int msm_hdmi_audio_update(struct hdmi *hdmi)
>>>>> return 0;
>>>>> }
>>>>> -int msm_hdmi_audio_info_setup(struct hdmi *hdmi, bool enabled,
>>>>> - uint32_t num_of_channels, uint32_t channel_allocation,
>>>>> - uint32_t level_shift, bool down_mix)
>>>>> +int msm_hdmi_audio_info_setup(struct hdmi *hdmi, int rate, int channels)
>>>>> {
>>>>> - struct hdmi_audio *audio;
>>>>> -
>>>>> if (!hdmi)
>>>>> return -ENXIO;
>>>>> - audio = &hdmi->audio;
>>>>> -
>>>>> - if (num_of_channels >= ARRAY_SIZE(nchannels))
>>>>> + if ((rate < 0) || (rate >= MSM_HDMI_SAMPLE_RATE_MAX))
>>>>> return -EINVAL;
>>>>> - audio->enabled = enabled;
>>>>> - audio->infoframe.channels = nchannels[num_of_channels];
>>>>> - audio->infoframe.channel_allocation = channel_allocation;
>>>>> - audio->infoframe.level_shift_value = level_shift;
>>>>> - audio->infoframe.downmix_inhibit = down_mix;
>>>>> + hdmi->audio.rate = rate;
>>>>> + hdmi->audio.channels = channels;
>>>>> + hdmi->audio.enabled = true;
>>>>> return msm_hdmi_audio_update(hdmi);
>>>>> }
>>>>> -void msm_hdmi_audio_set_sample_rate(struct hdmi *hdmi, int rate)
>>>>> +int msm_hdmi_audio_disable(struct hdmi *hdmi)
>>>>> {
>>>>> - struct hdmi_audio *audio;
>>>>> -
>>>>> if (!hdmi)
>>>>> - return;
>>>>> -
>>>>> - audio = &hdmi->audio;
>>>>> + return -ENXIO;
>>>>> - if ((rate < 0) || (rate >= MSM_HDMI_SAMPLE_RATE_MAX))
>>>>> - return;
>>>>> + hdmi->audio.rate = 0;
>>>>> + hdmi->audio.channels = 2;
>>>>> + hdmi->audio.enabled = false;
>>>>> - audio->rate = rate;
>>>>> - msm_hdmi_audio_update(hdmi);
>>>>> + return msm_hdmi_audio_update(hdmi);
>>>>> }
>>>>> diff --git a/drivers/gpu/drm/msm/hdmi/hdmi_bridge.c b/drivers/gpu/drm/msm/hdmi/hdmi_bridge.c
>>>>> index d5ab1f74c0e6f47dc59872c016104e9a84d85e9e..168b4104e705e8217f5d7ca5f902d7557c55ae24 100644
>>>>> --- a/drivers/gpu/drm/msm/hdmi/hdmi_bridge.c
>>>>> +++ b/drivers/gpu/drm/msm/hdmi/hdmi_bridge.c
>>>>> @@ -7,6 +7,8 @@
>>>>> #include <linux/delay.h>
>>>>> #include <drm/drm_bridge_connector.h>
>>>>> #include <drm/drm_edid.h>
>>>>> +#include <drm/display/drm_hdmi_helper.h>
>>>>> +#include <drm/display/drm_hdmi_state_helper.h>
>>>>> #include "msm_kms.h"
>>>>> #include "hdmi.h"
>>>>> @@ -68,23 +70,17 @@ static void power_off(struct drm_bridge *bridge)
>>>>> #define AVI_IFRAME_LINE_NUMBER 1
>>>>> -static void msm_hdmi_config_avi_infoframe(struct hdmi *hdmi)
>>>>> +static int msm_hdmi_config_avi_infoframe(struct hdmi *hdmi,
>>>>> + const u8 *buffer, size_t len)
>>>>> {
>>>>> - struct drm_crtc *crtc = hdmi->encoder->crtc;
>>>>> - const struct drm_display_mode *mode = &crtc->state->adjusted_mode;
>>>>> - union hdmi_infoframe frame;
>>>>> - u8 buffer[HDMI_INFOFRAME_SIZE(AVI)];
>>>>> + u32 buf[4] = {};
>>>>> u32 val;
>>>>> - int len;
>>>>> + int i;
>>>>> - drm_hdmi_avi_infoframe_from_display_mode(&frame.avi,
>>>>> - hdmi->connector, mode);
>>>>> -
>>>>> - len = hdmi_infoframe_pack(&frame, buffer, sizeof(buffer));
>>>>> - if (len < 0) {
>>>>> + if (len != HDMI_INFOFRAME_SIZE(AVI) || len - 3 > sizeof(buf)) {
>>>>> DRM_DEV_ERROR(&hdmi->pdev->dev,
>>>>> "failed to configure avi infoframe\n");
>>>>> - return;
>>>>> + return -EINVAL;
>>>>> }
>>>>> /*
>>>>> @@ -93,37 +89,118 @@ static void msm_hdmi_config_avi_infoframe(struct hdmi *hdmi)
>>>>> * written to the LSB byte of AVI_INFO0 and the version is written to
>>>>> * the third byte from the LSB of AVI_INFO3
>>>>> */
>>>>> - hdmi_write(hdmi, REG_HDMI_AVI_INFO(0),
>>>>> + memcpy(buf, &buffer[3], len - 3);
>>>>> +
>>>>> + buf[3] |= buffer[1] << 24;
>>>>> +
>>>>> + for (i = 0; i < ARRAY_SIZE(buf); i++)
>>>>> + hdmi_write(hdmi, REG_HDMI_AVI_INFO(i), buf[i]);
>>>>> +
>>>>> + val = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL1);
>>>>> + val |= HDMI_INFOFRAME_CTRL0_AVI_SEND |
>>>>> + HDMI_INFOFRAME_CTRL0_AVI_CONT;
>>>>> + hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL0, val);
>>>>> +
>>>>> + val = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL1);
>>>>> + val &= ~HDMI_INFOFRAME_CTRL1_AVI_INFO_LINE__MASK;
>>>>> + val |= HDMI_INFOFRAME_CTRL1_AVI_INFO_LINE(AVI_IFRAME_LINE_NUMBER);
>>>>> + hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL1, val);
>>>>> +
>>>>> + return 0;
>>>>> +}
>>>>> +
>>>>> +static int msm_hdmi_config_audio_infoframe(struct hdmi *hdmi,
>>>>> + const u8 *buffer, size_t len)
>>>>> +{
>>>>> + u32 val;
>>>>> +
>>>>> + if (len != HDMI_INFOFRAME_SIZE(AUDIO)) {
>>>>> + DRM_DEV_ERROR(&hdmi->pdev->dev,
>>>>> + "failed to configure audio infoframe\n");
>>>>> + return -EINVAL;
>>>>> + }
>>>>> +
>>>>> + hdmi_write(hdmi, REG_HDMI_AUDIO_INFO0,
>>>>> buffer[3] |
>>>>> buffer[4] << 8 |
>>>>> buffer[5] << 16 |
>>>>> buffer[6] << 24);
>>>>> - hdmi_write(hdmi, REG_HDMI_AVI_INFO(1),
>>>>> + hdmi_write(hdmi, REG_HDMI_AUDIO_INFO1,
>>>>> buffer[7] |
>>>>> buffer[8] << 8 |
>>>>> buffer[9] << 16 |
>>>>> buffer[10] << 24);
>>>>> - hdmi_write(hdmi, REG_HDMI_AVI_INFO(2),
>>>>> - buffer[11] |
>>>>> - buffer[12] << 8 |
>>>>> - buffer[13] << 16 |
>>>>> - buffer[14] << 24);
>>>>> + val = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL1);
>>>>> + val |= HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SEND |
>>>>> + HDMI_INFOFRAME_CTRL0_AUDIO_INFO_CONT |
>>>>> + HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SOURCE |
>>>>> + HDMI_INFOFRAME_CTRL0_AUDIO_INFO_UPDATE;
>>>>> + hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL0, val);
>>>>> +
>>>>> + return 0;
>>>>> +}
>>>>> +
>>>>> +static int msm_hdmi_bridge_clear_infoframe(struct drm_bridge *bridge,
>>>>> + enum hdmi_infoframe_type type)
>>>>> +{
>>>>> + struct hdmi_bridge *hdmi_bridge = to_hdmi_bridge(bridge);
>>>>> + struct hdmi *hdmi = hdmi_bridge->hdmi;
>>>>> + u32 val;
>>>>> +
>>>>> + switch (type) {
>>>>> + case HDMI_INFOFRAME_TYPE_AVI:
>>>>> + val = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL0);
>>>>> + val &= ~(HDMI_INFOFRAME_CTRL0_AVI_SEND |
>>>>> + HDMI_INFOFRAME_CTRL0_AVI_CONT);
>>>>> + hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL0, val);
>>>>> - hdmi_write(hdmi, REG_HDMI_AVI_INFO(3),
>>>>> - buffer[15] |
>>>>> - buffer[16] << 8 |
>>>>> - buffer[1] << 24);
>>>>> + val = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL1);
>>>>> + val &= ~HDMI_INFOFRAME_CTRL1_AVI_INFO_LINE__MASK;
>>>>> + hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL1, val);
>>>>> - hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL0,
>>>>> - HDMI_INFOFRAME_CTRL0_AVI_SEND |
>>>>> - HDMI_INFOFRAME_CTRL0_AVI_CONT);
>>>>> + break;
>>>>> - val = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL1);
>>>>> - val &= ~HDMI_INFOFRAME_CTRL1_AVI_INFO_LINE__MASK;
>>>>> - val |= HDMI_INFOFRAME_CTRL1_AVI_INFO_LINE(AVI_IFRAME_LINE_NUMBER);
>>>>> - hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL1, val);
>>>>> + case HDMI_INFOFRAME_TYPE_AUDIO:
>>>>> + val = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL0);
>>>>> + val &= ~(HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SEND |
>>>>> + HDMI_INFOFRAME_CTRL0_AUDIO_INFO_CONT |
>>>>> + HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SOURCE |
>>>>> + HDMI_INFOFRAME_CTRL0_AUDIO_INFO_UPDATE);
>>>>> + hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL0, val);
>>>>> +
>>>>> + val = hdmi_read(hdmi, REG_HDMI_INFOFRAME_CTRL1);
>>>>> + val &= ~HDMI_INFOFRAME_CTRL1_AUDIO_INFO_LINE__MASK;
>>>>> + hdmi_write(hdmi, REG_HDMI_INFOFRAME_CTRL1, val);
>>>>> +
>>>>> + break;
>>>>> +
>>>>> + default:
>>>>> + drm_dbg_driver(hdmi_bridge->base.dev, "Unsupported infoframe type %x\n", type);
>>>>> + }
>>>>> +
>>>>> + return 0;
>>>>> +}
>>>>> +
>>>>> +static int msm_hdmi_bridge_write_infoframe(struct drm_bridge *bridge,
>>>>> + enum hdmi_infoframe_type type,
>>>>> + const u8 *buffer, size_t len)
>>>>> +{
>>>>> + struct hdmi_bridge *hdmi_bridge = to_hdmi_bridge(bridge);
>>>>> + struct hdmi *hdmi = hdmi_bridge->hdmi;
>>>>> +
>>>>> + msm_hdmi_bridge_clear_infoframe(bridge, type);
>>>>> +
>>>>> + switch (type) {
>>>>> + case HDMI_INFOFRAME_TYPE_AVI:
>>>>> + return msm_hdmi_config_avi_infoframe(hdmi, buffer, len);
>>>>> + case HDMI_INFOFRAME_TYPE_AUDIO:
>>>>> + return msm_hdmi_config_audio_infoframe(hdmi, buffer, len);
>>>>> + default:
>>>>> + drm_dbg_driver(hdmi_bridge->base.dev, "Unsupported infoframe type %x\n", type);
>>>>> + return 0;
>>>>> + }
>>>>> }
>>>>> static void msm_hdmi_bridge_atomic_set_timings(struct hdmi *hdmi,
>>>>> @@ -146,16 +223,16 @@ static void msm_hdmi_bridge_atomic_pre_enable(struct drm_bridge *bridge,
>>>>> conn_state = drm_atomic_get_new_connector_state(state, connector);
>>>>> crtc_state = drm_atomic_get_new_crtc_state(state, conn_state->crtc);
>>>>> + hdmi->pixclock = conn_state->hdmi.tmds_char_rate;
>>>>> +
>>>>> if (!hdmi->power_on) {
>>>>> msm_hdmi_phy_resource_enable(phy);
>>>>> msm_hdmi_power_on(bridge);
>>>>> hdmi->power_on = true;
>>>>> - if (hdmi->hdmi_mode) {
>>>>> - msm_hdmi_config_avi_infoframe(hdmi);
>>>>> - msm_hdmi_audio_update(hdmi);
>>>>> - }
>>>>> }
>>>>> + drm_atomic_helper_connector_hdmi_update_infoframes(connector, state);
>>>>> +
>>>>> msm_hdmi_phy_powerup(phy, hdmi->pixclock);
>>>>> msm_hdmi_set_mode(hdmi, true);
>>>>> @@ -184,8 +261,6 @@ static void msm_hdmi_bridge_atomic_post_disable(struct drm_bridge *bridge,
>>>>> if (hdmi->power_on) {
>>>>> power_off(bridge);
>>>>> hdmi->power_on = false;
>>>>> - if (hdmi->hdmi_mode)
>>>>> - msm_hdmi_audio_update(hdmi);
>>>>> msm_hdmi_phy_resource_disable(phy);
>>>>> }
>>>>> }
>>>>> @@ -196,8 +271,6 @@ static void msm_hdmi_bridge_atomic_set_timings(struct hdmi *hdmi,
>>>>> int hstart, hend, vstart, vend;
>>>>> uint32_t frame_ctrl;
>>>>> - hdmi->pixclock = mode->clock * 1000;
>>>>> -
>>>>> hstart = mode->htotal - mode->hsync_start;
>>>>> hend = mode->htotal - mode->hsync_start + mode->hdisplay;
>>>>> @@ -241,9 +314,6 @@ static void msm_hdmi_bridge_atomic_set_timings(struct hdmi *hdmi,
>>>>> frame_ctrl |= HDMI_FRAME_CTRL_INTERLACED_EN;
>>>>> DBG("frame_ctrl=%08x", frame_ctrl);
>>>>> hdmi_write(hdmi, REG_HDMI_FRAME_CTRL, frame_ctrl);
>>>>> -
>>>>> - if (hdmi->hdmi_mode)
>>>>> - msm_hdmi_audio_update(hdmi);
>>>>> }
>>
>> Overall, I would like to know how the sequence was broken up:
>>
>> HDMI_INFOFRAME_CTRL0_AUDIO_INFO_SEND/CONT/UPDATE was moved from
>> msm_hdmi_audio_update() to msm_hdmi_config_audio_infoframe() which is
>> involed by drm_atomic_helper_connector_hdmi_update_infoframes() /
>> drm_atomic_helper_connector_hdmi_clear_audio_infoframe().
>>
>> But,HDMI_AUDIO_PKT_CTRL1_AUDIO_SAMPLE_SEND bit of REG_HDMI_AUDIO_PKT_CTRL1
>> remained in msm_hdmi_audio_update().
>
> This is correct, this bit controls sending of audio data, not the Audio
> InfoFrame.
>
> The strategy is very simple: Audio InfoFames are controlled centrally
> via the DRM HDMI framework. Correct InfoFrame data is programmed at the
> atomic_pre_enable() time (if it was set before) or during
> msm_hdmi_bridge_audio_prepare() when the new stream is started.
>
> All audio data frame management is deferred to
> msm_hdmi_bridge_audio_prepare().
>
Thats in the last patch of the series not this one. But I understand the
split.
>> If this is correct, are we not missing msm_hdmi_audio_update() in some
>> places like pre_enable() / post_disable()?
>
> drm_atomic_helper_connector_hdmi_update_infoframes() takes care of
> writing all InfoFrames, including the Audio one.
>
>>
>> Was this done because those calls were anyway bailing out audio->enabled was
>> not set by then?
>>
>> This seems to be another change which could have been done by itself to drop
>> those calls first and then make this change to make it more clear.
>>
>> OR atleast please explain these things better in the commit text.
>
> If the above text is fine to you, I'll add it to the commit message.
>
This is fine, but the dropping of msm_hdmi_audio_update() from
msm_hdmi_bridge_atomic_pre_enable() seems unrelated to the split OR
using the DRM HDMI framework. It seems more because that call by itself
without audio->enabled which is set in msm_hdmi_audio_info_setup() will
not do anything.
So this is actually an independent fix and was just combined with this
change? Thats the part I am trying to be more explicit about.
>
Powered by blists - more mailing lists