Thanks to advanced RE of fglrx we finally know what exactly needs to be
handled of AFMT change.

This has been tested for possible regressions on:
1) DCE2 HD2400 (RV610)
2) DCE3 HD3470 (RV620)

For a reference and details see:
https://bugzilla.kernel.org/show_bug.cgi?id=76231

Signed-off-by: Rafa? Mi?ecki <zajec5 at gmail.com>
---
V2: Move r600d.h changes to 0003 (from 0004) to don't break compilation
    Fix typo in commit message
---
 drivers/gpu/drm/radeon/r600_hdmi.c | 82 ++++++++++----------------------------
 1 file changed, 22 insertions(+), 60 deletions(-)

diff --git a/drivers/gpu/drm/radeon/r600_hdmi.c 
b/drivers/gpu/drm/radeon/r600_hdmi.c
index a611dd4..91f0e07 100644
--- a/drivers/gpu/drm/radeon/r600_hdmi.c
+++ b/drivers/gpu/drm/radeon/r600_hdmi.c
@@ -441,14 +441,16 @@ void r600_hdmi_setmode(struct drm_encoder *encoder, 
struct drm_display_mode *mod
        WREG32(HDMI0_RAMP_CONTROL2 + offset, 0x00000001);
        WREG32(HDMI0_RAMP_CONTROL3 + offset, 0x00000001);

-       r600_hdmi_audio_workaround(encoder);
-
        /* enable audio after to setting up hw */
        r600_audio_enable(rdev, dig->afmt->pin, true);
 }

-/*
- * update settings with current parameters from audio engine
+/**
+ * r600_hdmi_update_audio_settings - Update audio infoframe
+ *
+ * @encoder: drm encoder
+ *
+ * Gets info about current audio stream and updates audio infoframe.
  */
 void r600_hdmi_update_audio_settings(struct drm_encoder *encoder)
 {
@@ -460,7 +462,7 @@ void r600_hdmi_update_audio_settings(struct drm_encoder 
*encoder)
        uint8_t buffer[HDMI_INFOFRAME_HEADER_SIZE + HDMI_AUDIO_INFOFRAME_SIZE];
        struct hdmi_audio_infoframe frame;
        uint32_t offset;
-       uint32_t iec;
+       uint32_t value;
        ssize_t err;

        if (!dig->afmt || !dig->afmt->enabled)
@@ -473,60 +475,6 @@ void r600_hdmi_update_audio_settings(struct drm_encoder 
*encoder)
        DRM_DEBUG("0x%02X IEC60958 status bits and 0x%02X category code\n",
                  (int)audio.status_bits, (int)audio.category_code);

-       iec = 0;
-       if (audio.status_bits & AUDIO_STATUS_PROFESSIONAL)
-               iec |= 1 << 0;
-       if (audio.status_bits & AUDIO_STATUS_NONAUDIO)
-               iec |= 1 << 1;
-       if (audio.status_bits & AUDIO_STATUS_COPYRIGHT)
-               iec |= 1 << 2;
-       if (audio.status_bits & AUDIO_STATUS_EMPHASIS)
-               iec |= 1 << 3;
-
-       iec |= HDMI0_60958_CS_CATEGORY_CODE(audio.category_code);
-
-       switch (audio.rate) {
-       case 32000:
-               iec |= HDMI0_60958_CS_SAMPLING_FREQUENCY(0x3);
-               break;
-       case 44100:
-               iec |= HDMI0_60958_CS_SAMPLING_FREQUENCY(0x0);
-               break;
-       case 48000:
-               iec |= HDMI0_60958_CS_SAMPLING_FREQUENCY(0x2);
-               break;
-       case 88200:
-               iec |= HDMI0_60958_CS_SAMPLING_FREQUENCY(0x8);
-               break;
-       case 96000:
-               iec |= HDMI0_60958_CS_SAMPLING_FREQUENCY(0xa);
-               break;
-       case 176400:
-               iec |= HDMI0_60958_CS_SAMPLING_FREQUENCY(0xc);
-               break;
-       case 192000:
-               iec |= HDMI0_60958_CS_SAMPLING_FREQUENCY(0xe);
-               break;
-       }
-
-       WREG32(HDMI0_60958_0 + offset, iec);
-
-       iec = 0;
-       switch (audio.bits_per_sample) {
-       case 16:
-               iec |= HDMI0_60958_CS_WORD_LENGTH(0x2);
-               break;
-       case 20:
-               iec |= HDMI0_60958_CS_WORD_LENGTH(0x3);
-               break;
-       case 24:
-               iec |= HDMI0_60958_CS_WORD_LENGTH(0xb);
-               break;
-       }
-       if (audio.status_bits & AUDIO_STATUS_V)
-               iec |= 0x5 << 16;
-       WREG32_P(HDMI0_60958_1 + offset, iec, ~0x5000f);
-
        err = hdmi_audio_infoframe_init(&frame);
        if (err < 0) {
                DRM_ERROR("failed to setup audio infoframe\n");
@@ -541,8 +489,22 @@ void r600_hdmi_update_audio_settings(struct drm_encoder 
*encoder)
                return;
        }

+       value = RREG32(HDMI0_AUDIO_PACKET_CONTROL + offset);
+       if (value & HDMI0_AUDIO_TEST_EN)
+               WREG32(HDMI0_AUDIO_PACKET_CONTROL + offset,
+                      value & ~HDMI0_AUDIO_TEST_EN);
+
+       WREG32_OR(HDMI0_CONTROL + offset,
+                 HDMI0_ERROR_ACK);
+
+       WREG32_AND(HDMI0_INFOFRAME_CONTROL0 + offset,
+                  ~HDMI0_AUDIO_INFO_SOURCE);
+
        r600_hdmi_update_audio_infoframe(encoder, buffer, sizeof(buffer));
-       r600_hdmi_audio_workaround(encoder);
+
+       WREG32_OR(HDMI0_INFOFRAME_CONTROL0 + offset,
+                 HDMI0_AUDIO_INFO_CONT |
+                 HDMI0_AUDIO_INFO_UPDATE);
 }

 /*
-- 
1.8.4.5

Reply via email to