xref: /openbmc/linux/drivers/gpu/drm/radeon/dce6_afmt.c (revision b1880258)
1b530602fSAlex Deucher /*
2b530602fSAlex Deucher  * Copyright 2013 Advanced Micro Devices, Inc.
3b530602fSAlex Deucher  *
4b530602fSAlex Deucher  * Permission is hereby granted, free of charge, to any person obtaining a
5b530602fSAlex Deucher  * copy of this software and associated documentation files (the "Software"),
6b530602fSAlex Deucher  * to deal in the Software without restriction, including without limitation
7b530602fSAlex Deucher  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8b530602fSAlex Deucher  * and/or sell copies of the Software, and to permit persons to whom the
9b530602fSAlex Deucher  * Software is furnished to do so, subject to the following conditions:
10b530602fSAlex Deucher  *
11b530602fSAlex Deucher  * The above copyright notice and this permission notice shall be included in
12b530602fSAlex Deucher  * all copies or substantial portions of the Software.
13b530602fSAlex Deucher  *
14b530602fSAlex Deucher  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15b530602fSAlex Deucher  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16b530602fSAlex Deucher  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17b530602fSAlex Deucher  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18b530602fSAlex Deucher  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19b530602fSAlex Deucher  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20b530602fSAlex Deucher  * OTHER DEALINGS IN THE SOFTWARE.
21b530602fSAlex Deucher  *
22b530602fSAlex Deucher  */
23b530602fSAlex Deucher #include <linux/hdmi.h>
24b530602fSAlex Deucher #include <drm/drmP.h>
25b530602fSAlex Deucher #include "radeon.h"
26b530602fSAlex Deucher #include "sid.h"
27b530602fSAlex Deucher 
28b530602fSAlex Deucher static u32 dce6_endpoint_rreg(struct radeon_device *rdev,
29b530602fSAlex Deucher 			      u32 block_offset, u32 reg)
30b530602fSAlex Deucher {
310a5b7b0bSAlex Deucher 	unsigned long flags;
32b530602fSAlex Deucher 	u32 r;
33b530602fSAlex Deucher 
340a5b7b0bSAlex Deucher 	spin_lock_irqsave(&rdev->end_idx_lock, flags);
35b530602fSAlex Deucher 	WREG32(AZ_F0_CODEC_ENDPOINT_INDEX + block_offset, reg);
36b530602fSAlex Deucher 	r = RREG32(AZ_F0_CODEC_ENDPOINT_DATA + block_offset);
370a5b7b0bSAlex Deucher 	spin_unlock_irqrestore(&rdev->end_idx_lock, flags);
380a5b7b0bSAlex Deucher 
39b530602fSAlex Deucher 	return r;
40b530602fSAlex Deucher }
41b530602fSAlex Deucher 
42b530602fSAlex Deucher static void dce6_endpoint_wreg(struct radeon_device *rdev,
43b530602fSAlex Deucher 			       u32 block_offset, u32 reg, u32 v)
44b530602fSAlex Deucher {
450a5b7b0bSAlex Deucher 	unsigned long flags;
460a5b7b0bSAlex Deucher 
470a5b7b0bSAlex Deucher 	spin_lock_irqsave(&rdev->end_idx_lock, flags);
48b530602fSAlex Deucher 	if (ASIC_IS_DCE8(rdev))
49b530602fSAlex Deucher 		WREG32(AZ_F0_CODEC_ENDPOINT_INDEX + block_offset, reg);
50b530602fSAlex Deucher 	else
51b530602fSAlex Deucher 		WREG32(AZ_F0_CODEC_ENDPOINT_INDEX + block_offset,
52b530602fSAlex Deucher 		       AZ_ENDPOINT_REG_WRITE_EN | AZ_ENDPOINT_REG_INDEX(reg));
53b530602fSAlex Deucher 	WREG32(AZ_F0_CODEC_ENDPOINT_DATA + block_offset, v);
540a5b7b0bSAlex Deucher 	spin_unlock_irqrestore(&rdev->end_idx_lock, flags);
55b530602fSAlex Deucher }
56b530602fSAlex Deucher 
57b530602fSAlex Deucher #define RREG32_ENDPOINT(block, reg) dce6_endpoint_rreg(rdev, (block), (reg))
58b530602fSAlex Deucher #define WREG32_ENDPOINT(block, reg, v) dce6_endpoint_wreg(rdev, (block), (reg), (v))
59b530602fSAlex Deucher 
60b530602fSAlex Deucher 
61b530602fSAlex Deucher static void dce6_afmt_get_connected_pins(struct radeon_device *rdev)
62b530602fSAlex Deucher {
63b530602fSAlex Deucher 	int i;
64b530602fSAlex Deucher 	u32 offset, tmp;
65b530602fSAlex Deucher 
66b530602fSAlex Deucher 	for (i = 0; i < rdev->audio.num_pins; i++) {
67b530602fSAlex Deucher 		offset = rdev->audio.pin[i].offset;
68b530602fSAlex Deucher 		tmp = RREG32_ENDPOINT(offset,
69b530602fSAlex Deucher 				      AZ_F0_CODEC_PIN_CONTROL_RESPONSE_CONFIGURATION_DEFAULT);
70b530602fSAlex Deucher 		if (((tmp & PORT_CONNECTIVITY_MASK) >> PORT_CONNECTIVITY_SHIFT) == 1)
71b530602fSAlex Deucher 			rdev->audio.pin[i].connected = false;
72b530602fSAlex Deucher 		else
73b530602fSAlex Deucher 			rdev->audio.pin[i].connected = true;
74b530602fSAlex Deucher 	}
75b530602fSAlex Deucher }
76b530602fSAlex Deucher 
77b530602fSAlex Deucher struct r600_audio_pin *dce6_audio_get_pin(struct radeon_device *rdev)
78b530602fSAlex Deucher {
79b530602fSAlex Deucher 	int i;
80b530602fSAlex Deucher 
81b530602fSAlex Deucher 	dce6_afmt_get_connected_pins(rdev);
82b530602fSAlex Deucher 
83b530602fSAlex Deucher 	for (i = 0; i < rdev->audio.num_pins; i++) {
84b530602fSAlex Deucher 		if (rdev->audio.pin[i].connected)
85b530602fSAlex Deucher 			return &rdev->audio.pin[i];
86b530602fSAlex Deucher 	}
87b530602fSAlex Deucher 	DRM_ERROR("No connected audio pins found!\n");
88b530602fSAlex Deucher 	return NULL;
89b530602fSAlex Deucher }
90b530602fSAlex Deucher 
91b530602fSAlex Deucher void dce6_afmt_select_pin(struct drm_encoder *encoder)
92b530602fSAlex Deucher {
93b530602fSAlex Deucher 	struct radeon_device *rdev = encoder->dev->dev_private;
94b530602fSAlex Deucher 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
95b530602fSAlex Deucher 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
96b530602fSAlex Deucher 	u32 offset = dig->afmt->offset;
97b530602fSAlex Deucher 
98b530602fSAlex Deucher 	if (!dig->afmt->pin)
99b530602fSAlex Deucher 		return;
100b530602fSAlex Deucher 
1017cc0a3d8SAlex Deucher 	WREG32(AFMT_AUDIO_SRC_CONTROL + offset,
1027cc0a3d8SAlex Deucher 	       AFMT_AUDIO_SRC_SELECT(dig->afmt->pin->id));
103b530602fSAlex Deucher }
104b530602fSAlex Deucher 
105b1880258SAlex Deucher void dce6_afmt_write_latency_fields(struct drm_encoder *encoder,
106b1880258SAlex Deucher 				    struct drm_display_mode *mode)
107b1880258SAlex Deucher {
108b1880258SAlex Deucher 	struct radeon_device *rdev = encoder->dev->dev_private;
109b1880258SAlex Deucher 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
110b1880258SAlex Deucher 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
111b1880258SAlex Deucher 	struct drm_connector *connector;
112b1880258SAlex Deucher 	struct radeon_connector *radeon_connector = NULL;
113b1880258SAlex Deucher 	u32 tmp = 0, offset;
114b1880258SAlex Deucher 
115b1880258SAlex Deucher 	if (!dig->afmt->pin)
116b1880258SAlex Deucher 		return;
117b1880258SAlex Deucher 
118b1880258SAlex Deucher 	offset = dig->afmt->pin->offset;
119b1880258SAlex Deucher 
120b1880258SAlex Deucher 	list_for_each_entry(connector, &encoder->dev->mode_config.connector_list, head) {
121b1880258SAlex Deucher 		if (connector->encoder == encoder) {
122b1880258SAlex Deucher 			radeon_connector = to_radeon_connector(connector);
123b1880258SAlex Deucher 			break;
124b1880258SAlex Deucher 		}
125b1880258SAlex Deucher 	}
126b1880258SAlex Deucher 
127b1880258SAlex Deucher 	if (!radeon_connector) {
128b1880258SAlex Deucher 		DRM_ERROR("Couldn't find encoder's connector\n");
129b1880258SAlex Deucher 		return;
130b1880258SAlex Deucher 	}
131b1880258SAlex Deucher 
132b1880258SAlex Deucher 	if (mode->flags & DRM_MODE_FLAG_INTERLACE) {
133b1880258SAlex Deucher 		if (connector->latency_present[1])
134b1880258SAlex Deucher 			tmp = VIDEO_LIPSYNC(connector->video_latency[1]) |
135b1880258SAlex Deucher 				AUDIO_LIPSYNC(connector->audio_latency[1]);
136b1880258SAlex Deucher 		else
137b1880258SAlex Deucher 			tmp = VIDEO_LIPSYNC(255) | AUDIO_LIPSYNC(255);
138b1880258SAlex Deucher 	} else {
139b1880258SAlex Deucher 		if (connector->latency_present[0])
140b1880258SAlex Deucher 			tmp = VIDEO_LIPSYNC(connector->video_latency[0]) |
141b1880258SAlex Deucher 				AUDIO_LIPSYNC(connector->audio_latency[0]);
142b1880258SAlex Deucher 		else
143b1880258SAlex Deucher 			tmp = VIDEO_LIPSYNC(255) | AUDIO_LIPSYNC(255);
144b1880258SAlex Deucher 	}
145b1880258SAlex Deucher 	WREG32_ENDPOINT(offset, AZ_F0_CODEC_PIN_CONTROL_RESPONSE_LIPSYNC, tmp);
146b1880258SAlex Deucher }
147b1880258SAlex Deucher 
1486159b65aSRafał Miłecki void dce6_afmt_write_speaker_allocation(struct drm_encoder *encoder)
1496159b65aSRafał Miłecki {
1506159b65aSRafał Miłecki 	struct radeon_device *rdev = encoder->dev->dev_private;
1516159b65aSRafał Miłecki 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
1526159b65aSRafał Miłecki 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
1536159b65aSRafał Miłecki 	struct drm_connector *connector;
1546159b65aSRafał Miłecki 	struct radeon_connector *radeon_connector = NULL;
1556159b65aSRafał Miłecki 	u32 offset, tmp;
1566159b65aSRafał Miłecki 	u8 *sadb;
1576159b65aSRafał Miłecki 	int sad_count;
1586159b65aSRafał Miłecki 
1596159b65aSRafał Miłecki 	if (!dig->afmt->pin)
1606159b65aSRafał Miłecki 		return;
1616159b65aSRafał Miłecki 
1626159b65aSRafał Miłecki 	offset = dig->afmt->pin->offset;
1636159b65aSRafał Miłecki 
1646159b65aSRafał Miłecki 	list_for_each_entry(connector, &encoder->dev->mode_config.connector_list, head) {
1656159b65aSRafał Miłecki 		if (connector->encoder == encoder)
1666159b65aSRafał Miłecki 			radeon_connector = to_radeon_connector(connector);
1676159b65aSRafał Miłecki 	}
1686159b65aSRafał Miłecki 
1696159b65aSRafał Miłecki 	if (!radeon_connector) {
1706159b65aSRafał Miłecki 		DRM_ERROR("Couldn't find encoder's connector\n");
1716159b65aSRafał Miłecki 		return;
1726159b65aSRafał Miłecki 	}
1736159b65aSRafał Miłecki 
1746159b65aSRafał Miłecki 	sad_count = drm_edid_to_speaker_allocation(radeon_connector->edid, &sadb);
1756159b65aSRafał Miłecki 	if (sad_count < 0) {
1766159b65aSRafał Miłecki 		DRM_ERROR("Couldn't read Speaker Allocation Data Block: %d\n", sad_count);
1776159b65aSRafał Miłecki 		return;
1786159b65aSRafał Miłecki 	}
1796159b65aSRafał Miłecki 
1806159b65aSRafał Miłecki 	/* program the speaker allocation */
1816159b65aSRafał Miłecki 	tmp = RREG32_ENDPOINT(offset, AZ_F0_CODEC_PIN_CONTROL_CHANNEL_SPEAKER);
1826159b65aSRafał Miłecki 	tmp &= ~(DP_CONNECTION | SPEAKER_ALLOCATION_MASK);
1836159b65aSRafał Miłecki 	/* set HDMI mode */
1846159b65aSRafał Miłecki 	tmp |= HDMI_CONNECTION;
1856159b65aSRafał Miłecki 	if (sad_count)
1866159b65aSRafał Miłecki 		tmp |= SPEAKER_ALLOCATION(sadb[0]);
1876159b65aSRafał Miłecki 	else
1886159b65aSRafał Miłecki 		tmp |= SPEAKER_ALLOCATION(5); /* stereo */
1896159b65aSRafał Miłecki 	WREG32_ENDPOINT(offset, AZ_F0_CODEC_PIN_CONTROL_CHANNEL_SPEAKER, tmp);
1906159b65aSRafał Miłecki 
1916159b65aSRafał Miłecki 	kfree(sadb);
1926159b65aSRafał Miłecki }
1936159b65aSRafał Miłecki 
194b530602fSAlex Deucher void dce6_afmt_write_sad_regs(struct drm_encoder *encoder)
195b530602fSAlex Deucher {
196b530602fSAlex Deucher 	struct radeon_device *rdev = encoder->dev->dev_private;
197b530602fSAlex Deucher 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
198b530602fSAlex Deucher 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
1996159b65aSRafał Miłecki 	u32 offset;
200b530602fSAlex Deucher 	struct drm_connector *connector;
201b530602fSAlex Deucher 	struct radeon_connector *radeon_connector = NULL;
202b530602fSAlex Deucher 	struct cea_sad *sads;
2036159b65aSRafał Miłecki 	int i, sad_count;
204b530602fSAlex Deucher 
205b530602fSAlex Deucher 	static const u16 eld_reg_to_type[][2] = {
206b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR0, HDMI_AUDIO_CODING_TYPE_PCM },
207b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR1, HDMI_AUDIO_CODING_TYPE_AC3 },
208b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR2, HDMI_AUDIO_CODING_TYPE_MPEG1 },
209b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR3, HDMI_AUDIO_CODING_TYPE_MP3 },
210b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR4, HDMI_AUDIO_CODING_TYPE_MPEG2 },
211b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR5, HDMI_AUDIO_CODING_TYPE_AAC_LC },
212b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR6, HDMI_AUDIO_CODING_TYPE_DTS },
213b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR7, HDMI_AUDIO_CODING_TYPE_ATRAC },
214b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR9, HDMI_AUDIO_CODING_TYPE_EAC3 },
215b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR10, HDMI_AUDIO_CODING_TYPE_DTS_HD },
216b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR11, HDMI_AUDIO_CODING_TYPE_MLP },
217b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR13, HDMI_AUDIO_CODING_TYPE_WMA_PRO },
218b530602fSAlex Deucher 	};
219b530602fSAlex Deucher 
220b530602fSAlex Deucher 	if (!dig->afmt->pin)
221b530602fSAlex Deucher 		return;
222b530602fSAlex Deucher 
223b530602fSAlex Deucher 	offset = dig->afmt->pin->offset;
224b530602fSAlex Deucher 
225b530602fSAlex Deucher 	list_for_each_entry(connector, &encoder->dev->mode_config.connector_list, head) {
226b530602fSAlex Deucher 		if (connector->encoder == encoder)
227b530602fSAlex Deucher 			radeon_connector = to_radeon_connector(connector);
228b530602fSAlex Deucher 	}
229b530602fSAlex Deucher 
230b530602fSAlex Deucher 	if (!radeon_connector) {
231b530602fSAlex Deucher 		DRM_ERROR("Couldn't find encoder's connector\n");
232b530602fSAlex Deucher 		return;
233b530602fSAlex Deucher 	}
234b530602fSAlex Deucher 
235b530602fSAlex Deucher 	sad_count = drm_edid_to_sad(radeon_connector->edid, &sads);
236b530602fSAlex Deucher 	if (sad_count < 0) {
237b530602fSAlex Deucher 		DRM_ERROR("Couldn't read SADs: %d\n", sad_count);
238b530602fSAlex Deucher 		return;
239b530602fSAlex Deucher 	}
240b530602fSAlex Deucher 	BUG_ON(!sads);
241b530602fSAlex Deucher 
242b530602fSAlex Deucher 	for (i = 0; i < ARRAY_SIZE(eld_reg_to_type); i++) {
243b530602fSAlex Deucher 		u32 value = 0;
244b530602fSAlex Deucher 		int j;
245b530602fSAlex Deucher 
246b530602fSAlex Deucher 		for (j = 0; j < sad_count; j++) {
247b530602fSAlex Deucher 			struct cea_sad *sad = &sads[j];
248b530602fSAlex Deucher 
249b530602fSAlex Deucher 			if (sad->format == eld_reg_to_type[i][1]) {
250b530602fSAlex Deucher 				value = MAX_CHANNELS(sad->channels) |
251b530602fSAlex Deucher 					DESCRIPTOR_BYTE_2(sad->byte2) |
252b530602fSAlex Deucher 					SUPPORTED_FREQUENCIES(sad->freq);
253b530602fSAlex Deucher 				if (sad->format == HDMI_AUDIO_CODING_TYPE_PCM)
254b530602fSAlex Deucher 					value |= SUPPORTED_FREQUENCIES_STEREO(sad->freq);
255b530602fSAlex Deucher 				break;
256b530602fSAlex Deucher 			}
257b530602fSAlex Deucher 		}
258b530602fSAlex Deucher 		WREG32_ENDPOINT(offset, eld_reg_to_type[i][0], value);
259b530602fSAlex Deucher 	}
260b530602fSAlex Deucher 
261b530602fSAlex Deucher 	kfree(sads);
262b530602fSAlex Deucher }
263b530602fSAlex Deucher 
264b530602fSAlex Deucher static int dce6_audio_chipset_supported(struct radeon_device *rdev)
265b530602fSAlex Deucher {
266b530602fSAlex Deucher 	return !ASIC_IS_NODCE(rdev);
267b530602fSAlex Deucher }
268b530602fSAlex Deucher 
269b530602fSAlex Deucher static void dce6_audio_enable(struct radeon_device *rdev,
270b530602fSAlex Deucher 			      struct r600_audio_pin *pin,
271b530602fSAlex Deucher 			      bool enable)
272b530602fSAlex Deucher {
273b530602fSAlex Deucher 	WREG32_ENDPOINT(pin->offset, AZ_F0_CODEC_PIN_CONTROL_HOTPLUG_CONTROL,
274b530602fSAlex Deucher 			AUDIO_ENABLED);
275b530602fSAlex Deucher 	DRM_INFO("%s audio %d support\n", enable ? "Enabling" : "Disabling", pin->id);
276b530602fSAlex Deucher }
277b530602fSAlex Deucher 
278b530602fSAlex Deucher static const u32 pin_offsets[7] =
279b530602fSAlex Deucher {
280b530602fSAlex Deucher 	(0x5e00 - 0x5e00),
281b530602fSAlex Deucher 	(0x5e18 - 0x5e00),
282b530602fSAlex Deucher 	(0x5e30 - 0x5e00),
283b530602fSAlex Deucher 	(0x5e48 - 0x5e00),
284b530602fSAlex Deucher 	(0x5e60 - 0x5e00),
285b530602fSAlex Deucher 	(0x5e78 - 0x5e00),
286b530602fSAlex Deucher 	(0x5e90 - 0x5e00),
287b530602fSAlex Deucher };
288b530602fSAlex Deucher 
289b530602fSAlex Deucher int dce6_audio_init(struct radeon_device *rdev)
290b530602fSAlex Deucher {
291b530602fSAlex Deucher 	int i;
292b530602fSAlex Deucher 
293b530602fSAlex Deucher 	if (!radeon_audio || !dce6_audio_chipset_supported(rdev))
294b530602fSAlex Deucher 		return 0;
295b530602fSAlex Deucher 
296b530602fSAlex Deucher 	rdev->audio.enabled = true;
297b530602fSAlex Deucher 
298b530602fSAlex Deucher 	if (ASIC_IS_DCE8(rdev))
299b530602fSAlex Deucher 		rdev->audio.num_pins = 7;
300b530602fSAlex Deucher 	else
301b530602fSAlex Deucher 		rdev->audio.num_pins = 6;
302b530602fSAlex Deucher 
303b530602fSAlex Deucher 	for (i = 0; i < rdev->audio.num_pins; i++) {
304b530602fSAlex Deucher 		rdev->audio.pin[i].channels = -1;
305b530602fSAlex Deucher 		rdev->audio.pin[i].rate = -1;
306b530602fSAlex Deucher 		rdev->audio.pin[i].bits_per_sample = -1;
307b530602fSAlex Deucher 		rdev->audio.pin[i].status_bits = 0;
308b530602fSAlex Deucher 		rdev->audio.pin[i].category_code = 0;
309b530602fSAlex Deucher 		rdev->audio.pin[i].connected = false;
310b530602fSAlex Deucher 		rdev->audio.pin[i].offset = pin_offsets[i];
311b530602fSAlex Deucher 		rdev->audio.pin[i].id = i;
312b530602fSAlex Deucher 		dce6_audio_enable(rdev, &rdev->audio.pin[i], true);
313b530602fSAlex Deucher 	}
314b530602fSAlex Deucher 
315b530602fSAlex Deucher 	return 0;
316b530602fSAlex Deucher }
317b530602fSAlex Deucher 
318b530602fSAlex Deucher void dce6_audio_fini(struct radeon_device *rdev)
319b530602fSAlex Deucher {
320b530602fSAlex Deucher 	int i;
321b530602fSAlex Deucher 
322b530602fSAlex Deucher 	if (!rdev->audio.enabled)
323b530602fSAlex Deucher 		return;
324b530602fSAlex Deucher 
325b530602fSAlex Deucher 	for (i = 0; i < rdev->audio.num_pins; i++)
326b530602fSAlex Deucher 		dce6_audio_enable(rdev, &rdev->audio.pin[i], false);
327b530602fSAlex Deucher 
328b530602fSAlex Deucher 	rdev->audio.enabled = false;
329b530602fSAlex Deucher }
330