ALSA: hda: Conditionally use snooping for AMD HDMI
The recent regression report revealed that the use of WC pages for AMD
HDMI device together with AMD IOMMU leads to unexpected truncation or
noises. The issue seems triggered by the change in the kernel core
memory allocation that enables IOMMU driver to use always S/G
buffers. Meanwhile, the use of WC pages has been a workaround for the
similar issue with standard pages in the past. So, now we need to
apply the workaround conditionally, namely, only when IOMMU isn't in
place.
This patch modifies the workaround code to check the DMA ops at first
and apply the snoop-off only when needed.
Fixes: f5ff79fddf
("dma-mapping: remove CONFIG_DMA_REMAP")
Link: https://bugzilla.kernel.org/show_bug.cgi?id=219087
Link: https://patch.msgid.link/20240731170521.31714-1-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
parent
b7b7e1ab76
commit
478689b599
sound/pci/hda
@ -28,7 +28,7 @@
|
|||||||
#else
|
#else
|
||||||
#define AZX_DCAPS_I915_COMPONENT 0 /* NOP */
|
#define AZX_DCAPS_I915_COMPONENT 0 /* NOP */
|
||||||
#endif
|
#endif
|
||||||
/* 14 unused */
|
#define AZX_DCAPS_AMD_ALLOC_FIX (1 << 14) /* AMD allocation workaround */
|
||||||
#define AZX_DCAPS_CTX_WORKAROUND (1 << 15) /* X-Fi workaround */
|
#define AZX_DCAPS_CTX_WORKAROUND (1 << 15) /* X-Fi workaround */
|
||||||
#define AZX_DCAPS_POSFIX_LPIB (1 << 16) /* Use LPIB as default */
|
#define AZX_DCAPS_POSFIX_LPIB (1 << 16) /* Use LPIB as default */
|
||||||
#define AZX_DCAPS_AMD_WORKAROUND (1 << 17) /* AMD-specific workaround */
|
#define AZX_DCAPS_AMD_WORKAROUND (1 << 17) /* AMD-specific workaround */
|
||||||
|
@ -40,6 +40,7 @@
|
|||||||
|
|
||||||
#ifdef CONFIG_X86
|
#ifdef CONFIG_X86
|
||||||
/* for snoop control */
|
/* for snoop control */
|
||||||
|
#include <linux/dma-map-ops.h>
|
||||||
#include <asm/set_memory.h>
|
#include <asm/set_memory.h>
|
||||||
#include <asm/cpufeature.h>
|
#include <asm/cpufeature.h>
|
||||||
#endif
|
#endif
|
||||||
@ -306,7 +307,7 @@ enum {
|
|||||||
|
|
||||||
/* quirks for ATI HDMI with snoop off */
|
/* quirks for ATI HDMI with snoop off */
|
||||||
#define AZX_DCAPS_PRESET_ATI_HDMI_NS \
|
#define AZX_DCAPS_PRESET_ATI_HDMI_NS \
|
||||||
(AZX_DCAPS_PRESET_ATI_HDMI | AZX_DCAPS_SNOOP_OFF)
|
(AZX_DCAPS_PRESET_ATI_HDMI | AZX_DCAPS_AMD_ALLOC_FIX)
|
||||||
|
|
||||||
/* quirks for AMD SB */
|
/* quirks for AMD SB */
|
||||||
#define AZX_DCAPS_PRESET_AMD_SB \
|
#define AZX_DCAPS_PRESET_AMD_SB \
|
||||||
@ -1702,6 +1703,13 @@ static void azx_check_snoop_available(struct azx *chip)
|
|||||||
if (chip->driver_caps & AZX_DCAPS_SNOOP_OFF)
|
if (chip->driver_caps & AZX_DCAPS_SNOOP_OFF)
|
||||||
snoop = false;
|
snoop = false;
|
||||||
|
|
||||||
|
#ifdef CONFIG_X86
|
||||||
|
/* check the presence of DMA ops (i.e. IOMMU), disable snoop conditionally */
|
||||||
|
if ((chip->driver_caps & AZX_DCAPS_AMD_ALLOC_FIX) &&
|
||||||
|
!get_dma_ops(chip->card->dev))
|
||||||
|
snoop = false;
|
||||||
|
#endif
|
||||||
|
|
||||||
chip->snoop = snoop;
|
chip->snoop = snoop;
|
||||||
if (!snoop) {
|
if (!snoop) {
|
||||||
dev_info(chip->card->dev, "Force to non-snoop mode\n");
|
dev_info(chip->card->dev, "Force to non-snoop mode\n");
|
||||||
|
Loading…
Reference in New Issue
Block a user