提交 95b46c74 编写于 作者: N Neil Armstrong 提交者: Zheng Zengkai

drm/bridge: dw-hdmi: handle ELD when DRM_BRIDGE_ATTACH_NO_CONNECTOR

stable inclusion
from stable-v5.10.94
commit e3ba02b043f2fc8cd55f7f64d258b2efd70cdd4c
bugzilla: https://gitee.com/openeuler/kernel/issues/I531X9

Reference: https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git/commit/?id=e3ba02b043f2fc8cd55f7f64d258b2efd70cdd4c

--------------------------------

[ Upstream commit 3f2532d6 ]

The current ELD handling takes the internal connector ELD buffer and
shares it to the I2S and AHB sub-driver.

But with DRM_BRIDGE_ATTACH_NO_CONNECTOR, the connector is created
elsewhere (or not), and an eventual connector is known only
if the bridge chain up to a connector is enabled.

The current dw-hdmi code gets the current connector from
atomic_enable() so use the already stored connector pointer and
replace the buffer pointer with a callback returning the current
connector ELD buffer.

Since a connector is not always available, either pass an empty
ELD to the alsa HDMI driver or don't call snd_pcm_hw_constraint_eld()
in AHB driver.
Reported-by: NMartin Blumenstingl <martin.blumenstingl@googlemail.com>
Signed-off-by: NNeil Armstrong <narmstrong@baylibre.com>
[narmstrong: fixed typo in commit log]
Acked-by: NJernej Skrabec <jernej.skrabec@gmail.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20211029135947.3022875-1-narmstrong@baylibre.comSigned-off-by: NSasha Levin <sashal@kernel.org>
Signed-off-by: NZheng Zengkai <zhengzengkai@huawei.com>
Acked-by: NXie XiuQi <xiexiuqi@huawei.com>
上级 ff646e36
...@@ -320,13 +320,17 @@ static int dw_hdmi_open(struct snd_pcm_substream *substream) ...@@ -320,13 +320,17 @@ static int dw_hdmi_open(struct snd_pcm_substream *substream)
struct snd_pcm_runtime *runtime = substream->runtime; struct snd_pcm_runtime *runtime = substream->runtime;
struct snd_dw_hdmi *dw = substream->private_data; struct snd_dw_hdmi *dw = substream->private_data;
void __iomem *base = dw->data.base; void __iomem *base = dw->data.base;
u8 *eld;
int ret; int ret;
runtime->hw = dw_hdmi_hw; runtime->hw = dw_hdmi_hw;
ret = snd_pcm_hw_constraint_eld(runtime, dw->data.eld); eld = dw->data.get_eld(dw->data.hdmi);
if (eld) {
ret = snd_pcm_hw_constraint_eld(runtime, eld);
if (ret < 0) if (ret < 0)
return ret; return ret;
}
ret = snd_pcm_limit_hw_rates(runtime); ret = snd_pcm_limit_hw_rates(runtime);
if (ret < 0) if (ret < 0)
......
...@@ -9,15 +9,15 @@ struct dw_hdmi_audio_data { ...@@ -9,15 +9,15 @@ struct dw_hdmi_audio_data {
void __iomem *base; void __iomem *base;
int irq; int irq;
struct dw_hdmi *hdmi; struct dw_hdmi *hdmi;
u8 *eld; u8 *(*get_eld)(struct dw_hdmi *hdmi);
}; };
struct dw_hdmi_i2s_audio_data { struct dw_hdmi_i2s_audio_data {
struct dw_hdmi *hdmi; struct dw_hdmi *hdmi;
u8 *eld;
void (*write)(struct dw_hdmi *hdmi, u8 val, int offset); void (*write)(struct dw_hdmi *hdmi, u8 val, int offset);
u8 (*read)(struct dw_hdmi *hdmi, int offset); u8 (*read)(struct dw_hdmi *hdmi, int offset);
u8 *(*get_eld)(struct dw_hdmi *hdmi);
}; };
#endif #endif
...@@ -135,8 +135,15 @@ static int dw_hdmi_i2s_get_eld(struct device *dev, void *data, uint8_t *buf, ...@@ -135,8 +135,15 @@ static int dw_hdmi_i2s_get_eld(struct device *dev, void *data, uint8_t *buf,
size_t len) size_t len)
{ {
struct dw_hdmi_i2s_audio_data *audio = data; struct dw_hdmi_i2s_audio_data *audio = data;
u8 *eld;
eld = audio->get_eld(audio->hdmi);
if (eld)
memcpy(buf, eld, min_t(size_t, MAX_ELD_BYTES, len));
else
/* Pass en empty ELD if connector not available */
memset(buf, 0, len);
memcpy(buf, audio->eld, min_t(size_t, MAX_ELD_BYTES, len));
return 0; return 0;
} }
......
...@@ -756,6 +756,14 @@ static void hdmi_enable_audio_clk(struct dw_hdmi *hdmi, bool enable) ...@@ -756,6 +756,14 @@ static void hdmi_enable_audio_clk(struct dw_hdmi *hdmi, bool enable)
hdmi_writeb(hdmi, hdmi->mc_clkdis, HDMI_MC_CLKDIS); hdmi_writeb(hdmi, hdmi->mc_clkdis, HDMI_MC_CLKDIS);
} }
static u8 *hdmi_audio_get_eld(struct dw_hdmi *hdmi)
{
if (!hdmi->curr_conn)
return NULL;
return hdmi->curr_conn->eld;
}
static void dw_hdmi_ahb_audio_enable(struct dw_hdmi *hdmi) static void dw_hdmi_ahb_audio_enable(struct dw_hdmi *hdmi)
{ {
hdmi_set_cts_n(hdmi, hdmi->audio_cts, hdmi->audio_n); hdmi_set_cts_n(hdmi, hdmi->audio_cts, hdmi->audio_n);
...@@ -3395,7 +3403,7 @@ struct dw_hdmi *dw_hdmi_probe(struct platform_device *pdev, ...@@ -3395,7 +3403,7 @@ struct dw_hdmi *dw_hdmi_probe(struct platform_device *pdev,
audio.base = hdmi->regs; audio.base = hdmi->regs;
audio.irq = irq; audio.irq = irq;
audio.hdmi = hdmi; audio.hdmi = hdmi;
audio.eld = hdmi->connector.eld; audio.get_eld = hdmi_audio_get_eld;
hdmi->enable_audio = dw_hdmi_ahb_audio_enable; hdmi->enable_audio = dw_hdmi_ahb_audio_enable;
hdmi->disable_audio = dw_hdmi_ahb_audio_disable; hdmi->disable_audio = dw_hdmi_ahb_audio_disable;
...@@ -3408,7 +3416,7 @@ struct dw_hdmi *dw_hdmi_probe(struct platform_device *pdev, ...@@ -3408,7 +3416,7 @@ struct dw_hdmi *dw_hdmi_probe(struct platform_device *pdev,
struct dw_hdmi_i2s_audio_data audio; struct dw_hdmi_i2s_audio_data audio;
audio.hdmi = hdmi; audio.hdmi = hdmi;
audio.eld = hdmi->connector.eld; audio.get_eld = hdmi_audio_get_eld;
audio.write = hdmi_writeb; audio.write = hdmi_writeb;
audio.read = hdmi_readb; audio.read = hdmi_readb;
hdmi->enable_audio = dw_hdmi_i2s_audio_enable; hdmi->enable_audio = dw_hdmi_i2s_audio_enable;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册