• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (c) 2016-2020, The Linux Foundation. All rights reserved.
4  */
5 
6 
7 #define pr_fmt(fmt)	"[drm-dp] %s: " fmt, __func__
8 
9 #include <linux/of_platform.h>
10 
11 #include <drm/drm_dp_helper.h>
12 #include <drm/drm_edid.h>
13 
14 #include "dp_catalog.h"
15 #include "dp_audio.h"
16 #include "dp_panel.h"
17 #include "dp_display.h"
18 
19 #define HEADER_BYTE_2_BIT	 0
20 #define PARITY_BYTE_2_BIT	 8
21 #define HEADER_BYTE_1_BIT	16
22 #define PARITY_BYTE_1_BIT	24
23 #define HEADER_BYTE_3_BIT	16
24 #define PARITY_BYTE_3_BIT	24
25 
26 struct dp_audio_private {
27 	struct platform_device *audio_pdev;
28 	struct platform_device *pdev;
29 	struct dp_catalog *catalog;
30 	struct dp_panel *panel;
31 
32 	bool engine_on;
33 	u32 channels;
34 
35 	struct dp_audio dp_audio;
36 };
37 
dp_audio_get_g0_value(u8 data)38 static u8 dp_audio_get_g0_value(u8 data)
39 {
40 	u8 c[4];
41 	u8 g[4];
42 	u8 ret_data = 0;
43 	u8 i;
44 
45 	for (i = 0; i < 4; i++)
46 		c[i] = (data >> i) & 0x01;
47 
48 	g[0] = c[3];
49 	g[1] = c[0] ^ c[3];
50 	g[2] = c[1];
51 	g[3] = c[2];
52 
53 	for (i = 0; i < 4; i++)
54 		ret_data = ((g[i] & 0x01) << i) | ret_data;
55 
56 	return ret_data;
57 }
58 
dp_audio_get_g1_value(u8 data)59 static u8 dp_audio_get_g1_value(u8 data)
60 {
61 	u8 c[4];
62 	u8 g[4];
63 	u8 ret_data = 0;
64 	u8 i;
65 
66 	for (i = 0; i < 4; i++)
67 		c[i] = (data >> i) & 0x01;
68 
69 	g[0] = c[0] ^ c[3];
70 	g[1] = c[0] ^ c[1] ^ c[3];
71 	g[2] = c[1] ^ c[2];
72 	g[3] = c[2] ^ c[3];
73 
74 	for (i = 0; i < 4; i++)
75 		ret_data = ((g[i] & 0x01) << i) | ret_data;
76 
77 	return ret_data;
78 }
79 
dp_audio_calculate_parity(u32 data)80 static u8 dp_audio_calculate_parity(u32 data)
81 {
82 	u8 x0 = 0;
83 	u8 x1 = 0;
84 	u8 ci = 0;
85 	u8 iData = 0;
86 	u8 i = 0;
87 	u8 parity_byte;
88 	u8 num_byte = (data & 0xFF00) > 0 ? 8 : 2;
89 
90 	for (i = 0; i < num_byte; i++) {
91 		iData = (data >> i*4) & 0xF;
92 
93 		ci = iData ^ x1;
94 		x1 = x0 ^ dp_audio_get_g1_value(ci);
95 		x0 = dp_audio_get_g0_value(ci);
96 	}
97 
98 	parity_byte = x1 | (x0 << 4);
99 
100 	return parity_byte;
101 }
102 
dp_audio_get_header(struct dp_catalog * catalog,enum dp_catalog_audio_sdp_type sdp,enum dp_catalog_audio_header_type header)103 static u32 dp_audio_get_header(struct dp_catalog *catalog,
104 		enum dp_catalog_audio_sdp_type sdp,
105 		enum dp_catalog_audio_header_type header)
106 {
107 	catalog->sdp_type = sdp;
108 	catalog->sdp_header = header;
109 	dp_catalog_audio_get_header(catalog);
110 
111 	return catalog->audio_data;
112 }
113 
dp_audio_set_header(struct dp_catalog * catalog,u32 data,enum dp_catalog_audio_sdp_type sdp,enum dp_catalog_audio_header_type header)114 static void dp_audio_set_header(struct dp_catalog *catalog,
115 		u32 data,
116 		enum dp_catalog_audio_sdp_type sdp,
117 		enum dp_catalog_audio_header_type header)
118 {
119 	catalog->sdp_type = sdp;
120 	catalog->sdp_header = header;
121 	catalog->audio_data = data;
122 	dp_catalog_audio_set_header(catalog);
123 }
124 
dp_audio_stream_sdp(struct dp_audio_private * audio)125 static void dp_audio_stream_sdp(struct dp_audio_private *audio)
126 {
127 	struct dp_catalog *catalog = audio->catalog;
128 	u32 value, new_value;
129 	u8 parity_byte;
130 
131 	/* Config header and parity byte 1 */
132 	value = dp_audio_get_header(catalog,
133 			DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_1);
134 
135 	new_value = 0x02;
136 	parity_byte = dp_audio_calculate_parity(new_value);
137 	value |= ((new_value << HEADER_BYTE_1_BIT)
138 			| (parity_byte << PARITY_BYTE_1_BIT));
139 	DRM_DEBUG_DP("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
140 			value, parity_byte);
141 	dp_audio_set_header(catalog, value,
142 		DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_1);
143 
144 	/* Config header and parity byte 2 */
145 	value = dp_audio_get_header(catalog,
146 			DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_2);
147 	new_value = value;
148 	parity_byte = dp_audio_calculate_parity(new_value);
149 	value |= ((new_value << HEADER_BYTE_2_BIT)
150 			| (parity_byte << PARITY_BYTE_2_BIT));
151 	DRM_DEBUG_DP("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
152 			value, parity_byte);
153 
154 	dp_audio_set_header(catalog, value,
155 		DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_2);
156 
157 	/* Config header and parity byte 3 */
158 	value = dp_audio_get_header(catalog,
159 			DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_3);
160 
161 	new_value = audio->channels - 1;
162 	parity_byte = dp_audio_calculate_parity(new_value);
163 	value |= ((new_value << HEADER_BYTE_3_BIT)
164 			| (parity_byte << PARITY_BYTE_3_BIT));
165 	DRM_DEBUG_DP("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
166 		value, parity_byte);
167 
168 	dp_audio_set_header(catalog, value,
169 		DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_3);
170 }
171 
dp_audio_timestamp_sdp(struct dp_audio_private * audio)172 static void dp_audio_timestamp_sdp(struct dp_audio_private *audio)
173 {
174 	struct dp_catalog *catalog = audio->catalog;
175 	u32 value, new_value;
176 	u8 parity_byte;
177 
178 	/* Config header and parity byte 1 */
179 	value = dp_audio_get_header(catalog,
180 			DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_1);
181 
182 	new_value = 0x1;
183 	parity_byte = dp_audio_calculate_parity(new_value);
184 	value |= ((new_value << HEADER_BYTE_1_BIT)
185 			| (parity_byte << PARITY_BYTE_1_BIT));
186 	DRM_DEBUG_DP("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
187 		value, parity_byte);
188 	dp_audio_set_header(catalog, value,
189 		DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_1);
190 
191 	/* Config header and parity byte 2 */
192 	value = dp_audio_get_header(catalog,
193 			DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_2);
194 
195 	new_value = 0x17;
196 	parity_byte = dp_audio_calculate_parity(new_value);
197 	value |= ((new_value << HEADER_BYTE_2_BIT)
198 			| (parity_byte << PARITY_BYTE_2_BIT));
199 	DRM_DEBUG_DP("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
200 			value, parity_byte);
201 	dp_audio_set_header(catalog, value,
202 		DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_2);
203 
204 	/* Config header and parity byte 3 */
205 	value = dp_audio_get_header(catalog,
206 			DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_3);
207 
208 	new_value = (0x0 | (0x11 << 2));
209 	parity_byte = dp_audio_calculate_parity(new_value);
210 	value |= ((new_value << HEADER_BYTE_3_BIT)
211 			| (parity_byte << PARITY_BYTE_3_BIT));
212 	DRM_DEBUG_DP("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
213 			value, parity_byte);
214 	dp_audio_set_header(catalog, value,
215 		DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_3);
216 }
217 
dp_audio_infoframe_sdp(struct dp_audio_private * audio)218 static void dp_audio_infoframe_sdp(struct dp_audio_private *audio)
219 {
220 	struct dp_catalog *catalog = audio->catalog;
221 	u32 value, new_value;
222 	u8 parity_byte;
223 
224 	/* Config header and parity byte 1 */
225 	value = dp_audio_get_header(catalog,
226 			DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_1);
227 
228 	new_value = 0x84;
229 	parity_byte = dp_audio_calculate_parity(new_value);
230 	value |= ((new_value << HEADER_BYTE_1_BIT)
231 			| (parity_byte << PARITY_BYTE_1_BIT));
232 	DRM_DEBUG_DP("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
233 			value, parity_byte);
234 	dp_audio_set_header(catalog, value,
235 		DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_1);
236 
237 	/* Config header and parity byte 2 */
238 	value = dp_audio_get_header(catalog,
239 			DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_2);
240 
241 	new_value = 0x1b;
242 	parity_byte = dp_audio_calculate_parity(new_value);
243 	value |= ((new_value << HEADER_BYTE_2_BIT)
244 			| (parity_byte << PARITY_BYTE_2_BIT));
245 	DRM_DEBUG_DP("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
246 			value, parity_byte);
247 	dp_audio_set_header(catalog, value,
248 		DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_2);
249 
250 	/* Config header and parity byte 3 */
251 	value = dp_audio_get_header(catalog,
252 			DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_3);
253 
254 	new_value = (0x0 | (0x11 << 2));
255 	parity_byte = dp_audio_calculate_parity(new_value);
256 	value |= ((new_value << HEADER_BYTE_3_BIT)
257 			| (parity_byte << PARITY_BYTE_3_BIT));
258 	DRM_DEBUG_DP("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
259 			new_value, parity_byte);
260 	dp_audio_set_header(catalog, value,
261 		DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_3);
262 }
263 
dp_audio_copy_management_sdp(struct dp_audio_private * audio)264 static void dp_audio_copy_management_sdp(struct dp_audio_private *audio)
265 {
266 	struct dp_catalog *catalog = audio->catalog;
267 	u32 value, new_value;
268 	u8 parity_byte;
269 
270 	/* Config header and parity byte 1 */
271 	value = dp_audio_get_header(catalog,
272 			DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_1);
273 
274 	new_value = 0x05;
275 	parity_byte = dp_audio_calculate_parity(new_value);
276 	value |= ((new_value << HEADER_BYTE_1_BIT)
277 			| (parity_byte << PARITY_BYTE_1_BIT));
278 	DRM_DEBUG_DP("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
279 			value, parity_byte);
280 	dp_audio_set_header(catalog, value,
281 		DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_1);
282 
283 	/* Config header and parity byte 2 */
284 	value = dp_audio_get_header(catalog,
285 			DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_2);
286 
287 	new_value = 0x0F;
288 	parity_byte = dp_audio_calculate_parity(new_value);
289 	value |= ((new_value << HEADER_BYTE_2_BIT)
290 			| (parity_byte << PARITY_BYTE_2_BIT));
291 	DRM_DEBUG_DP("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
292 			value, parity_byte);
293 	dp_audio_set_header(catalog, value,
294 		DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_2);
295 
296 	/* Config header and parity byte 3 */
297 	value = dp_audio_get_header(catalog,
298 			DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_3);
299 
300 	new_value = 0x0;
301 	parity_byte = dp_audio_calculate_parity(new_value);
302 	value |= ((new_value << HEADER_BYTE_3_BIT)
303 			| (parity_byte << PARITY_BYTE_3_BIT));
304 	DRM_DEBUG_DP("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
305 			value, parity_byte);
306 	dp_audio_set_header(catalog, value,
307 		DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_3);
308 }
309 
dp_audio_isrc_sdp(struct dp_audio_private * audio)310 static void dp_audio_isrc_sdp(struct dp_audio_private *audio)
311 {
312 	struct dp_catalog *catalog = audio->catalog;
313 	u32 value, new_value;
314 	u8 parity_byte;
315 
316 	/* Config header and parity byte 1 */
317 	value = dp_audio_get_header(catalog,
318 			DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_1);
319 
320 	new_value = 0x06;
321 	parity_byte = dp_audio_calculate_parity(new_value);
322 	value |= ((new_value << HEADER_BYTE_1_BIT)
323 			| (parity_byte << PARITY_BYTE_1_BIT));
324 	DRM_DEBUG_DP("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
325 			value, parity_byte);
326 	dp_audio_set_header(catalog, value,
327 		DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_1);
328 
329 	/* Config header and parity byte 2 */
330 	value = dp_audio_get_header(catalog,
331 			DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_2);
332 
333 	new_value = 0x0F;
334 	parity_byte = dp_audio_calculate_parity(new_value);
335 	value |= ((new_value << HEADER_BYTE_2_BIT)
336 			| (parity_byte << PARITY_BYTE_2_BIT));
337 	DRM_DEBUG_DP("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
338 			value, parity_byte);
339 	dp_audio_set_header(catalog, value,
340 		DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_2);
341 }
342 
dp_audio_setup_sdp(struct dp_audio_private * audio)343 static void dp_audio_setup_sdp(struct dp_audio_private *audio)
344 {
345 	dp_catalog_audio_config_sdp(audio->catalog);
346 
347 	dp_audio_stream_sdp(audio);
348 	dp_audio_timestamp_sdp(audio);
349 	dp_audio_infoframe_sdp(audio);
350 	dp_audio_copy_management_sdp(audio);
351 	dp_audio_isrc_sdp(audio);
352 }
353 
dp_audio_setup_acr(struct dp_audio_private * audio)354 static void dp_audio_setup_acr(struct dp_audio_private *audio)
355 {
356 	u32 select = 0;
357 	struct dp_catalog *catalog = audio->catalog;
358 
359 	switch (audio->dp_audio.bw_code) {
360 	case DP_LINK_BW_1_62:
361 		select = 0;
362 		break;
363 	case DP_LINK_BW_2_7:
364 		select = 1;
365 		break;
366 	case DP_LINK_BW_5_4:
367 		select = 2;
368 		break;
369 	case DP_LINK_BW_8_1:
370 		select = 3;
371 		break;
372 	default:
373 		DRM_DEBUG_DP("Unknown link rate\n");
374 		select = 0;
375 		break;
376 	}
377 
378 	catalog->audio_data = select;
379 	dp_catalog_audio_config_acr(catalog);
380 }
381 
dp_audio_safe_to_exit_level(struct dp_audio_private * audio)382 static void dp_audio_safe_to_exit_level(struct dp_audio_private *audio)
383 {
384 	struct dp_catalog *catalog = audio->catalog;
385 	u32 safe_to_exit_level = 0;
386 
387 	switch (audio->dp_audio.lane_count) {
388 	case 1:
389 		safe_to_exit_level = 14;
390 		break;
391 	case 2:
392 		safe_to_exit_level = 8;
393 		break;
394 	case 4:
395 		safe_to_exit_level = 5;
396 		break;
397 	default:
398 		DRM_DEBUG_DP("setting the default safe_to_exit_level = %u\n",
399 				safe_to_exit_level);
400 		safe_to_exit_level = 14;
401 		break;
402 	}
403 
404 	catalog->audio_data = safe_to_exit_level;
405 	dp_catalog_audio_sfe_level(catalog);
406 }
407 
dp_audio_enable(struct dp_audio_private * audio,bool enable)408 static void dp_audio_enable(struct dp_audio_private *audio, bool enable)
409 {
410 	struct dp_catalog *catalog = audio->catalog;
411 
412 	catalog->audio_data = enable;
413 	dp_catalog_audio_enable(catalog);
414 
415 	audio->engine_on = enable;
416 }
417 
dp_audio_get_data(struct platform_device * pdev)418 static struct dp_audio_private *dp_audio_get_data(struct platform_device *pdev)
419 {
420 	struct dp_audio *dp_audio;
421 	struct msm_dp *dp_display;
422 
423 	if (!pdev) {
424 		DRM_ERROR("invalid input\n");
425 		return ERR_PTR(-ENODEV);
426 	}
427 
428 	dp_display = platform_get_drvdata(pdev);
429 	if (!dp_display) {
430 		DRM_ERROR("invalid input\n");
431 		return ERR_PTR(-ENODEV);
432 	}
433 
434 	dp_audio = dp_display->dp_audio;
435 
436 	if (!dp_audio) {
437 		DRM_ERROR("invalid dp_audio data\n");
438 		return ERR_PTR(-EINVAL);
439 	}
440 
441 	return container_of(dp_audio, struct dp_audio_private, dp_audio);
442 }
443 
dp_audio_hook_plugged_cb(struct device * dev,void * data,hdmi_codec_plugged_cb fn,struct device * codec_dev)444 static int dp_audio_hook_plugged_cb(struct device *dev, void *data,
445 		hdmi_codec_plugged_cb fn,
446 		struct device *codec_dev)
447 {
448 
449 	struct platform_device *pdev;
450 	struct msm_dp *dp_display;
451 
452 	pdev = to_platform_device(dev);
453 	if (!pdev) {
454 		pr_err("invalid input\n");
455 		return -ENODEV;
456 	}
457 
458 	dp_display = platform_get_drvdata(pdev);
459 	if (!dp_display) {
460 		pr_err("invalid input\n");
461 		return -ENODEV;
462 	}
463 
464 	return dp_display_set_plugged_cb(dp_display, fn, codec_dev);
465 }
466 
dp_audio_get_eld(struct device * dev,void * data,uint8_t * buf,size_t len)467 static int dp_audio_get_eld(struct device *dev,
468 	void *data, uint8_t *buf, size_t len)
469 {
470 	struct platform_device *pdev;
471 	struct msm_dp *dp_display;
472 
473 	pdev = to_platform_device(dev);
474 
475 	if (!pdev) {
476 		DRM_ERROR("invalid input\n");
477 		return -ENODEV;
478 	}
479 
480 	dp_display = platform_get_drvdata(pdev);
481 	if (!dp_display) {
482 		DRM_ERROR("invalid input\n");
483 		return -ENODEV;
484 	}
485 
486 	memcpy(buf, dp_display->connector->eld,
487 		min(sizeof(dp_display->connector->eld), len));
488 
489 	return 0;
490 }
491 
dp_audio_hw_params(struct device * dev,void * data,struct hdmi_codec_daifmt * daifmt,struct hdmi_codec_params * params)492 int dp_audio_hw_params(struct device *dev,
493 	void *data,
494 	struct hdmi_codec_daifmt *daifmt,
495 	struct hdmi_codec_params *params)
496 {
497 	int rc = 0;
498 	struct dp_audio_private *audio;
499 	struct platform_device *pdev;
500 	struct msm_dp *dp_display;
501 
502 	pdev = to_platform_device(dev);
503 	dp_display = platform_get_drvdata(pdev);
504 
505 	/*
506 	 * there could be cases where sound card can be opened even
507 	 * before OR even when DP is not connected . This can cause
508 	 * unclocked access as the audio subsystem relies on the DP
509 	 * driver to maintain the correct state of clocks. To protect
510 	 * such cases check for connection status and bail out if not
511 	 * connected.
512 	 */
513 	if (!dp_display->power_on) {
514 		rc = -EINVAL;
515 		goto end;
516 	}
517 
518 	audio = dp_audio_get_data(pdev);
519 	if (IS_ERR(audio)) {
520 		rc = PTR_ERR(audio);
521 		goto end;
522 	}
523 
524 	audio->channels = params->channels;
525 
526 	dp_audio_setup_sdp(audio);
527 	dp_audio_setup_acr(audio);
528 	dp_audio_safe_to_exit_level(audio);
529 	dp_audio_enable(audio, true);
530 	dp_display_signal_audio_start(dp_display);
531 	dp_display->audio_enabled = true;
532 
533 end:
534 	return rc;
535 }
536 
dp_audio_shutdown(struct device * dev,void * data)537 static void dp_audio_shutdown(struct device *dev, void *data)
538 {
539 	struct dp_audio_private *audio;
540 	struct platform_device *pdev;
541 	struct msm_dp *dp_display;
542 
543 	pdev = to_platform_device(dev);
544 	dp_display = platform_get_drvdata(pdev);
545 	audio = dp_audio_get_data(pdev);
546 	if (IS_ERR(audio)) {
547 		DRM_ERROR("failed to get audio data\n");
548 		return;
549 	}
550 
551 	/*
552 	 * if audio was not enabled there is no need
553 	 * to execute the shutdown and we can bail out early.
554 	 * This also makes sure that we dont cause an unclocked
555 	 * access when audio subsystem calls this without DP being
556 	 * connected. is_connected cannot be used here as its set
557 	 * to false earlier than this call
558 	 */
559 	if (!dp_display->audio_enabled)
560 		return;
561 
562 	dp_audio_enable(audio, false);
563 	/* signal the dp display to safely shutdown clocks */
564 	dp_display_signal_audio_complete(dp_display);
565 }
566 
567 static const struct hdmi_codec_ops dp_audio_codec_ops = {
568 	.hw_params = dp_audio_hw_params,
569 	.audio_shutdown = dp_audio_shutdown,
570 	.get_eld = dp_audio_get_eld,
571 	.hook_plugged_cb = dp_audio_hook_plugged_cb,
572 };
573 
574 static struct hdmi_codec_pdata codec_data = {
575 	.ops = &dp_audio_codec_ops,
576 	.max_i2s_channels = 8,
577 	.i2s = 1,
578 };
579 
dp_register_audio_driver(struct device * dev,struct dp_audio * dp_audio)580 int dp_register_audio_driver(struct device *dev,
581 		struct dp_audio *dp_audio)
582 {
583 	struct dp_audio_private *audio_priv;
584 
585 	audio_priv = container_of(dp_audio,
586 			struct dp_audio_private, dp_audio);
587 
588 	audio_priv->audio_pdev = platform_device_register_data(dev,
589 						HDMI_CODEC_DRV_NAME,
590 						PLATFORM_DEVID_AUTO,
591 						&codec_data,
592 						sizeof(codec_data));
593 	return PTR_ERR_OR_ZERO(audio_priv->audio_pdev);
594 }
595 
dp_audio_get(struct platform_device * pdev,struct dp_panel * panel,struct dp_catalog * catalog)596 struct dp_audio *dp_audio_get(struct platform_device *pdev,
597 			struct dp_panel *panel,
598 			struct dp_catalog *catalog)
599 {
600 	int rc = 0;
601 	struct dp_audio_private *audio;
602 	struct dp_audio *dp_audio;
603 
604 	if (!pdev || !panel || !catalog) {
605 		DRM_ERROR("invalid input\n");
606 		rc = -EINVAL;
607 		goto error;
608 	}
609 
610 	audio = devm_kzalloc(&pdev->dev, sizeof(*audio), GFP_KERNEL);
611 	if (!audio) {
612 		rc = -ENOMEM;
613 		goto error;
614 	}
615 
616 	audio->pdev = pdev;
617 	audio->panel = panel;
618 	audio->catalog = catalog;
619 
620 	dp_audio = &audio->dp_audio;
621 
622 	dp_catalog_audio_init(catalog);
623 
624 	return dp_audio;
625 error:
626 	return ERR_PTR(rc);
627 }
628 
dp_audio_put(struct dp_audio * dp_audio)629 void dp_audio_put(struct dp_audio *dp_audio)
630 {
631 	struct dp_audio_private *audio;
632 
633 	if (!dp_audio)
634 		return;
635 
636 	audio = container_of(dp_audio, struct dp_audio_private, dp_audio);
637 
638 	devm_kfree(&audio->pdev->dev, audio);
639 }
640