xref: /openbmc/linux/drivers/gpu/drm/radeon/dce6_afmt.c (revision 7735a7c9)
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>
24c182615fSSam Ravnborg 
25*7735a7c9SLee Jones #include "dce6_afmt.h"
26b530602fSAlex Deucher #include "radeon.h"
271a626b68SSlava Grigorev #include "radeon_audio.h"
28b530602fSAlex Deucher #include "sid.h"
29b530602fSAlex Deucher 
302afa3265SSlava Grigorev #define DCE8_DCCG_AUDIO_DTO1_PHASE	0x05b8
312afa3265SSlava Grigorev #define DCE8_DCCG_AUDIO_DTO1_MODULE	0x05bc
322afa3265SSlava Grigorev 
dce6_endpoint_rreg(struct radeon_device * rdev,u32 block_offset,u32 reg)331a626b68SSlava Grigorev u32 dce6_endpoint_rreg(struct radeon_device *rdev,
34b530602fSAlex Deucher 			      u32 block_offset, u32 reg)
35b530602fSAlex Deucher {
360a5b7b0bSAlex Deucher 	unsigned long flags;
37b530602fSAlex Deucher 	u32 r;
38b530602fSAlex Deucher 
390a5b7b0bSAlex Deucher 	spin_lock_irqsave(&rdev->end_idx_lock, flags);
40b530602fSAlex Deucher 	WREG32(AZ_F0_CODEC_ENDPOINT_INDEX + block_offset, reg);
41b530602fSAlex Deucher 	r = RREG32(AZ_F0_CODEC_ENDPOINT_DATA + block_offset);
420a5b7b0bSAlex Deucher 	spin_unlock_irqrestore(&rdev->end_idx_lock, flags);
430a5b7b0bSAlex Deucher 
44b530602fSAlex Deucher 	return r;
45b530602fSAlex Deucher }
46b530602fSAlex Deucher 
dce6_endpoint_wreg(struct radeon_device * rdev,u32 block_offset,u32 reg,u32 v)471a626b68SSlava Grigorev void dce6_endpoint_wreg(struct radeon_device *rdev,
48b530602fSAlex Deucher 			       u32 block_offset, u32 reg, u32 v)
49b530602fSAlex Deucher {
500a5b7b0bSAlex Deucher 	unsigned long flags;
510a5b7b0bSAlex Deucher 
520a5b7b0bSAlex Deucher 	spin_lock_irqsave(&rdev->end_idx_lock, flags);
53b530602fSAlex Deucher 	if (ASIC_IS_DCE8(rdev))
54b530602fSAlex Deucher 		WREG32(AZ_F0_CODEC_ENDPOINT_INDEX + block_offset, reg);
55b530602fSAlex Deucher 	else
56b530602fSAlex Deucher 		WREG32(AZ_F0_CODEC_ENDPOINT_INDEX + block_offset,
57b530602fSAlex Deucher 		       AZ_ENDPOINT_REG_WRITE_EN | AZ_ENDPOINT_REG_INDEX(reg));
58b530602fSAlex Deucher 	WREG32(AZ_F0_CODEC_ENDPOINT_DATA + block_offset, v);
590a5b7b0bSAlex Deucher 	spin_unlock_irqrestore(&rdev->end_idx_lock, flags);
60b530602fSAlex Deucher }
61b530602fSAlex Deucher 
dce6_afmt_get_connected_pins(struct radeon_device * rdev)62b530602fSAlex Deucher static void dce6_afmt_get_connected_pins(struct radeon_device *rdev)
63b530602fSAlex Deucher {
64b530602fSAlex Deucher 	int i;
65b530602fSAlex Deucher 	u32 offset, tmp;
66b530602fSAlex Deucher 
67b530602fSAlex Deucher 	for (i = 0; i < rdev->audio.num_pins; i++) {
68b530602fSAlex Deucher 		offset = rdev->audio.pin[i].offset;
69b530602fSAlex Deucher 		tmp = RREG32_ENDPOINT(offset,
70b530602fSAlex Deucher 				      AZ_F0_CODEC_PIN_CONTROL_RESPONSE_CONFIGURATION_DEFAULT);
71b530602fSAlex Deucher 		if (((tmp & PORT_CONNECTIVITY_MASK) >> PORT_CONNECTIVITY_SHIFT) == 1)
72b530602fSAlex Deucher 			rdev->audio.pin[i].connected = false;
73b530602fSAlex Deucher 		else
74b530602fSAlex Deucher 			rdev->audio.pin[i].connected = true;
75b530602fSAlex Deucher 	}
76b530602fSAlex Deucher }
77b530602fSAlex Deucher 
dce6_audio_get_pin(struct radeon_device * rdev)78b530602fSAlex Deucher struct r600_audio_pin *dce6_audio_get_pin(struct radeon_device *rdev)
79b530602fSAlex Deucher {
80d0465208SAlex Deucher 	struct drm_encoder *encoder;
81d0465208SAlex Deucher 	struct radeon_encoder *radeon_encoder;
82d0465208SAlex Deucher 	struct radeon_encoder_atom_dig *dig;
83d0465208SAlex Deucher 	struct r600_audio_pin *pin = NULL;
84d0465208SAlex Deucher 	int i, pin_count;
85b530602fSAlex Deucher 
86b530602fSAlex Deucher 	dce6_afmt_get_connected_pins(rdev);
87b530602fSAlex Deucher 
88b530602fSAlex Deucher 	for (i = 0; i < rdev->audio.num_pins; i++) {
89d0465208SAlex Deucher 		if (rdev->audio.pin[i].connected) {
90d0465208SAlex Deucher 			pin = &rdev->audio.pin[i];
91d0465208SAlex Deucher 			pin_count = 0;
92d0465208SAlex Deucher 
93d0465208SAlex Deucher 			list_for_each_entry(encoder, &rdev->ddev->mode_config.encoder_list, head) {
94d0465208SAlex Deucher 				if (radeon_encoder_is_digital(encoder)) {
95d0465208SAlex Deucher 					radeon_encoder = to_radeon_encoder(encoder);
96d0465208SAlex Deucher 					dig = radeon_encoder->enc_priv;
97d0465208SAlex Deucher 					if (dig->pin == pin)
98d0465208SAlex Deucher 						pin_count++;
99b530602fSAlex Deucher 				}
100d0465208SAlex Deucher 			}
101d0465208SAlex Deucher 
102d0465208SAlex Deucher 			if (pin_count == 0)
103d0465208SAlex Deucher 				return pin;
104d0465208SAlex Deucher 		}
105d0465208SAlex Deucher 	}
106d0465208SAlex Deucher 	if (!pin)
107b530602fSAlex Deucher 		DRM_ERROR("No connected audio pins found!\n");
108d0465208SAlex Deucher 	return pin;
109b530602fSAlex Deucher }
110b530602fSAlex Deucher 
dce6_afmt_select_pin(struct drm_encoder * encoder)111b530602fSAlex Deucher void dce6_afmt_select_pin(struct drm_encoder *encoder)
112b530602fSAlex Deucher {
113b530602fSAlex Deucher 	struct radeon_device *rdev = encoder->dev->dev_private;
114b530602fSAlex Deucher 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
115b530602fSAlex Deucher 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
116b530602fSAlex Deucher 
117d0ea397eSAlex Deucher 	if (!dig || !dig->afmt || !dig->pin)
118b530602fSAlex Deucher 		return;
119b530602fSAlex Deucher 
120d0ea397eSAlex Deucher 	WREG32(AFMT_AUDIO_SRC_CONTROL +  dig->afmt->offset,
121d0ea397eSAlex Deucher 	       AFMT_AUDIO_SRC_SELECT(dig->pin->id));
122b530602fSAlex Deucher }
123b530602fSAlex Deucher 
dce6_afmt_write_latency_fields(struct drm_encoder * encoder,struct drm_connector * connector,struct drm_display_mode * mode)124b1880258SAlex Deucher void dce6_afmt_write_latency_fields(struct drm_encoder *encoder,
125d0ea397eSAlex Deucher 				    struct drm_connector *connector,
126d0ea397eSAlex Deucher 				    struct drm_display_mode *mode)
127b1880258SAlex Deucher {
128b1880258SAlex Deucher 	struct radeon_device *rdev = encoder->dev->dev_private;
129b1880258SAlex Deucher 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
130b1880258SAlex Deucher 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
131d0ea397eSAlex Deucher 	u32 tmp = 0;
132b1880258SAlex Deucher 
133d0ea397eSAlex Deucher 	if (!dig || !dig->afmt || !dig->pin)
134b1880258SAlex Deucher 		return;
135b1880258SAlex Deucher 
136b1880258SAlex Deucher 	if (mode->flags & DRM_MODE_FLAG_INTERLACE) {
137b1880258SAlex Deucher 		if (connector->latency_present[1])
138b1880258SAlex Deucher 			tmp = VIDEO_LIPSYNC(connector->video_latency[1]) |
139b1880258SAlex Deucher 				AUDIO_LIPSYNC(connector->audio_latency[1]);
140b1880258SAlex Deucher 		else
141c748990bSStefan Brüns 			tmp = VIDEO_LIPSYNC(0) | AUDIO_LIPSYNC(0);
142b1880258SAlex Deucher 	} else {
143b1880258SAlex Deucher 		if (connector->latency_present[0])
144b1880258SAlex Deucher 			tmp = VIDEO_LIPSYNC(connector->video_latency[0]) |
145b1880258SAlex Deucher 				AUDIO_LIPSYNC(connector->audio_latency[0]);
146b1880258SAlex Deucher 		else
147c748990bSStefan Brüns 			tmp = VIDEO_LIPSYNC(0) | AUDIO_LIPSYNC(0);
148b1880258SAlex Deucher 	}
149d0ea397eSAlex Deucher 	WREG32_ENDPOINT(dig->pin->offset,
150d0ea397eSAlex Deucher 			AZ_F0_CODEC_PIN_CONTROL_RESPONSE_LIPSYNC, tmp);
151b1880258SAlex Deucher }
152b1880258SAlex Deucher 
dce6_afmt_hdmi_write_speaker_allocation(struct drm_encoder * encoder,u8 * sadb,int sad_count)15300a9d4bcSSlava Grigorev void dce6_afmt_hdmi_write_speaker_allocation(struct drm_encoder *encoder,
15400a9d4bcSSlava Grigorev 					     u8 *sadb, int sad_count)
1556159b65aSRafał Miłecki {
1566159b65aSRafał Miłecki 	struct radeon_device *rdev = encoder->dev->dev_private;
1576159b65aSRafał Miłecki 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
1586159b65aSRafał Miłecki 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
159d0ea397eSAlex Deucher 	u32 tmp;
1606159b65aSRafał Miłecki 
161d0ea397eSAlex Deucher 	if (!dig || !dig->afmt || !dig->pin)
1626159b65aSRafał Miłecki 		return;
1636159b65aSRafał Miłecki 
1646159b65aSRafał Miłecki 	/* program the speaker allocation */
165d0ea397eSAlex Deucher 	tmp = RREG32_ENDPOINT(dig->pin->offset,
166d0ea397eSAlex Deucher 			      AZ_F0_CODEC_PIN_CONTROL_CHANNEL_SPEAKER);
1676159b65aSRafał Miłecki 	tmp &= ~(DP_CONNECTION | SPEAKER_ALLOCATION_MASK);
1686159b65aSRafał Miłecki 	/* set HDMI mode */
1696159b65aSRafał Miłecki 	tmp |= HDMI_CONNECTION;
1706159b65aSRafał Miłecki 	if (sad_count)
1716159b65aSRafał Miłecki 		tmp |= SPEAKER_ALLOCATION(sadb[0]);
1726159b65aSRafał Miłecki 	else
1736159b65aSRafał Miłecki 		tmp |= SPEAKER_ALLOCATION(5); /* stereo */
174d0ea397eSAlex Deucher 	WREG32_ENDPOINT(dig->pin->offset,
175d0ea397eSAlex Deucher 			AZ_F0_CODEC_PIN_CONTROL_CHANNEL_SPEAKER, tmp);
17600a9d4bcSSlava Grigorev }
1776159b65aSRafał Miłecki 
dce6_afmt_dp_write_speaker_allocation(struct drm_encoder * encoder,u8 * sadb,int sad_count)17800a9d4bcSSlava Grigorev void dce6_afmt_dp_write_speaker_allocation(struct drm_encoder *encoder,
17900a9d4bcSSlava Grigorev 					   u8 *sadb, int sad_count)
18000a9d4bcSSlava Grigorev {
18100a9d4bcSSlava Grigorev 	struct radeon_device *rdev = encoder->dev->dev_private;
18200a9d4bcSSlava Grigorev 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
18300a9d4bcSSlava Grigorev 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
184d0ea397eSAlex Deucher 	u32 tmp;
18500a9d4bcSSlava Grigorev 
186d0ea397eSAlex Deucher 	if (!dig || !dig->afmt || !dig->pin)
18700a9d4bcSSlava Grigorev 		return;
18800a9d4bcSSlava Grigorev 
18900a9d4bcSSlava Grigorev 	/* program the speaker allocation */
190d0ea397eSAlex Deucher 	tmp = RREG32_ENDPOINT(dig->pin->offset,
191d0ea397eSAlex Deucher 			      AZ_F0_CODEC_PIN_CONTROL_CHANNEL_SPEAKER);
19200a9d4bcSSlava Grigorev 	tmp &= ~(HDMI_CONNECTION | SPEAKER_ALLOCATION_MASK);
19300a9d4bcSSlava Grigorev 	/* set DP mode */
19400a9d4bcSSlava Grigorev 	tmp |= DP_CONNECTION;
19500a9d4bcSSlava Grigorev 	if (sad_count)
19600a9d4bcSSlava Grigorev 		tmp |= SPEAKER_ALLOCATION(sadb[0]);
19700a9d4bcSSlava Grigorev 	else
19800a9d4bcSSlava Grigorev 		tmp |= SPEAKER_ALLOCATION(5); /* stereo */
199d0ea397eSAlex Deucher 	WREG32_ENDPOINT(dig->pin->offset,
200d0ea397eSAlex Deucher 			AZ_F0_CODEC_PIN_CONTROL_CHANNEL_SPEAKER, tmp);
2016159b65aSRafał Miłecki }
2026159b65aSRafał Miłecki 
dce6_afmt_write_sad_regs(struct drm_encoder * encoder,struct cea_sad * sads,int sad_count)203070a2e63SAlex Deucher void dce6_afmt_write_sad_regs(struct drm_encoder *encoder,
204070a2e63SAlex Deucher 			      struct cea_sad *sads, int sad_count)
205b530602fSAlex Deucher {
206070a2e63SAlex Deucher 	int i;
207b530602fSAlex Deucher 	struct radeon_encoder *radeon_encoder = to_radeon_encoder(encoder);
208b530602fSAlex Deucher 	struct radeon_encoder_atom_dig *dig = radeon_encoder->enc_priv;
209070a2e63SAlex Deucher 	struct radeon_device *rdev = encoder->dev->dev_private;
210b530602fSAlex Deucher 	static const u16 eld_reg_to_type[][2] = {
211b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR0, HDMI_AUDIO_CODING_TYPE_PCM },
212b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR1, HDMI_AUDIO_CODING_TYPE_AC3 },
213b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR2, HDMI_AUDIO_CODING_TYPE_MPEG1 },
214b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR3, HDMI_AUDIO_CODING_TYPE_MP3 },
215b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR4, HDMI_AUDIO_CODING_TYPE_MPEG2 },
216b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR5, HDMI_AUDIO_CODING_TYPE_AAC_LC },
217b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR6, HDMI_AUDIO_CODING_TYPE_DTS },
218b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR7, HDMI_AUDIO_CODING_TYPE_ATRAC },
219b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR9, HDMI_AUDIO_CODING_TYPE_EAC3 },
220b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR10, HDMI_AUDIO_CODING_TYPE_DTS_HD },
221b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR11, HDMI_AUDIO_CODING_TYPE_MLP },
222b530602fSAlex Deucher 		{ AZ_F0_CODEC_PIN_CONTROL_AUDIO_DESCRIPTOR13, HDMI_AUDIO_CODING_TYPE_WMA_PRO },
223b530602fSAlex Deucher 	};
224b530602fSAlex Deucher 
225d0ea397eSAlex Deucher 	if (!dig || !dig->afmt || !dig->pin)
226b530602fSAlex Deucher 		return;
227b530602fSAlex Deucher 
228b530602fSAlex Deucher 	for (i = 0; i < ARRAY_SIZE(eld_reg_to_type); i++) {
229b530602fSAlex Deucher 		u32 value = 0;
2300f57bca9SAnssi Hannula 		u8 stereo_freqs = 0;
2310f57bca9SAnssi Hannula 		int max_channels = -1;
232b530602fSAlex Deucher 		int j;
233b530602fSAlex Deucher 
234b530602fSAlex Deucher 		for (j = 0; j < sad_count; j++) {
235b530602fSAlex Deucher 			struct cea_sad *sad = &sads[j];
236b530602fSAlex Deucher 
237b530602fSAlex Deucher 			if (sad->format == eld_reg_to_type[i][1]) {
2380f57bca9SAnssi Hannula 				if (sad->channels > max_channels) {
239b530602fSAlex Deucher 					value = MAX_CHANNELS(sad->channels) |
240b530602fSAlex Deucher 						DESCRIPTOR_BYTE_2(sad->byte2) |
241b530602fSAlex Deucher 						SUPPORTED_FREQUENCIES(sad->freq);
2420f57bca9SAnssi Hannula 					max_channels = sad->channels;
2430f57bca9SAnssi Hannula 				}
2440f57bca9SAnssi Hannula 
245b530602fSAlex Deucher 				if (sad->format == HDMI_AUDIO_CODING_TYPE_PCM)
2460f57bca9SAnssi Hannula 					stereo_freqs |= sad->freq;
2470f57bca9SAnssi Hannula 				else
248b530602fSAlex Deucher 					break;
249b530602fSAlex Deucher 			}
250b530602fSAlex Deucher 		}
2510f57bca9SAnssi Hannula 
2520f57bca9SAnssi Hannula 		value |= SUPPORTED_FREQUENCIES_STEREO(stereo_freqs);
2530f57bca9SAnssi Hannula 
254d0ea397eSAlex Deucher 		WREG32_ENDPOINT(dig->pin->offset, eld_reg_to_type[i][0], value);
255b530602fSAlex Deucher 	}
256b530602fSAlex Deucher }
257b530602fSAlex Deucher 
dce6_audio_enable(struct radeon_device * rdev,struct r600_audio_pin * pin,u8 enable_mask)258832eafafSAlex Deucher void dce6_audio_enable(struct radeon_device *rdev,
259b530602fSAlex Deucher 		       struct r600_audio_pin *pin,
260d3d8c141SAlex Deucher 		       u8 enable_mask)
261b530602fSAlex Deucher {
262832eafafSAlex Deucher 	if (!pin)
263832eafafSAlex Deucher 		return;
264832eafafSAlex Deucher 
265f68fdbe4SAlex Deucher 	WREG32_ENDPOINT(pin->offset, AZ_F0_CODEC_PIN_CONTROL_HOT_PLUG_CONTROL,
266d3d8c141SAlex Deucher 			enable_mask ? AUDIO_ENABLED : 0);
267b530602fSAlex Deucher }
268a85d682aSSlava Grigorev 
dce6_hdmi_audio_set_dto(struct radeon_device * rdev,struct radeon_crtc * crtc,unsigned int clock)269a85d682aSSlava Grigorev void dce6_hdmi_audio_set_dto(struct radeon_device *rdev,
270a85d682aSSlava Grigorev 			     struct radeon_crtc *crtc, unsigned int clock)
271a85d682aSSlava Grigorev {
272a85d682aSSlava Grigorev 	/* Two dtos; generally use dto0 for HDMI */
273a85d682aSSlava Grigorev 	u32 value = 0;
274a85d682aSSlava Grigorev 
275a85d682aSSlava Grigorev 	if (crtc)
276a85d682aSSlava Grigorev 		value |= DCCG_AUDIO_DTO0_SOURCE_SEL(crtc->crtc_id);
277a85d682aSSlava Grigorev 
278a85d682aSSlava Grigorev 	WREG32(DCCG_AUDIO_DTO_SOURCE, value);
279a85d682aSSlava Grigorev 
280a85d682aSSlava Grigorev 	/* Express [24MHz / target pixel clock] as an exact rational
281a85d682aSSlava Grigorev 	 * number (coefficient of two integer numbers.  DCCG_AUDIO_DTOx_PHASE
282a85d682aSSlava Grigorev 	 * is the numerator, DCCG_AUDIO_DTOx_MODULE is the denominator
283a85d682aSSlava Grigorev 	 */
284a85d682aSSlava Grigorev 	WREG32(DCCG_AUDIO_DTO0_PHASE, 24000);
285a85d682aSSlava Grigorev 	WREG32(DCCG_AUDIO_DTO0_MODULE, clock);
286a85d682aSSlava Grigorev }
287a85d682aSSlava Grigorev 
dce6_dp_audio_set_dto(struct radeon_device * rdev,struct radeon_crtc * crtc,unsigned int clock)288a85d682aSSlava Grigorev void dce6_dp_audio_set_dto(struct radeon_device *rdev,
289a85d682aSSlava Grigorev 			   struct radeon_crtc *crtc, unsigned int clock)
290a85d682aSSlava Grigorev {
291a85d682aSSlava Grigorev 	/* Two dtos; generally use dto1 for DP */
292a85d682aSSlava Grigorev 	u32 value = 0;
293a85d682aSSlava Grigorev 	value |= DCCG_AUDIO_DTO_SEL;
294a85d682aSSlava Grigorev 
295a85d682aSSlava Grigorev 	if (crtc)
296a85d682aSSlava Grigorev 		value |= DCCG_AUDIO_DTO0_SOURCE_SEL(crtc->crtc_id);
297a85d682aSSlava Grigorev 
298a85d682aSSlava Grigorev 	WREG32(DCCG_AUDIO_DTO_SOURCE, value);
299a85d682aSSlava Grigorev 
300a85d682aSSlava Grigorev 	/* Express [24MHz / target pixel clock] as an exact rational
301a85d682aSSlava Grigorev 	 * number (coefficient of two integer numbers.  DCCG_AUDIO_DTOx_PHASE
302a85d682aSSlava Grigorev 	 * is the numerator, DCCG_AUDIO_DTOx_MODULE is the denominator
303a85d682aSSlava Grigorev 	 */
3042afa3265SSlava Grigorev 	if (ASIC_IS_DCE8(rdev)) {
305ac4a9350SSlava Grigorev 		unsigned int div = (RREG32(DENTIST_DISPCLK_CNTL) &
306ac4a9350SSlava Grigorev 			DENTIST_DPREFCLK_WDIVIDER_MASK) >>
307ac4a9350SSlava Grigorev 			DENTIST_DPREFCLK_WDIVIDER_SHIFT;
308a64c9dabSSlava Grigorev 		div = radeon_audio_decode_dfs_div(div);
309ac4a9350SSlava Grigorev 
310ac4a9350SSlava Grigorev 		if (div)
311a64c9dabSSlava Grigorev 			clock = clock * 100 / div;
312ac4a9350SSlava Grigorev 
3132afa3265SSlava Grigorev 		WREG32(DCE8_DCCG_AUDIO_DTO1_PHASE, 24000);
3142afa3265SSlava Grigorev 		WREG32(DCE8_DCCG_AUDIO_DTO1_MODULE, clock);
3152afa3265SSlava Grigorev 	} else {
316a85d682aSSlava Grigorev 		WREG32(DCCG_AUDIO_DTO1_PHASE, 24000);
317a85d682aSSlava Grigorev 		WREG32(DCCG_AUDIO_DTO1_MODULE, clock);
318a85d682aSSlava Grigorev 	}
3192afa3265SSlava Grigorev }
320