msm-dai-q6-hdmi-v2.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612
  1. /* Copyright (c) 2012-2018, The Linux Foundation. All rights reserved.
  2. *
  3. * This program is free software; you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License version 2 and
  5. * only version 2 as published by the Free Software Foundation.
  6. *
  7. * This program is distributed in the hope that it will be useful,
  8. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. * GNU General Public License for more details.
  11. */
  12. #include <linux/init.h>
  13. #include <linux/module.h>
  14. #include <linux/device.h>
  15. #include <linux/platform_device.h>
  16. #include <linux/bitops.h>
  17. #include <linux/slab.h>
  18. #include <linux/of_device.h>
  19. #include <sound/core.h>
  20. #include <sound/pcm.h>
  21. #include <sound/soc.h>
  22. #include <sound/pcm_params.h>
  23. #include <dsp/apr_audio-v2.h>
  24. #include <dsp/q6afe-v2.h>
  25. #include "msm-dai-q6-v2.h"
  26. #define HDMI_RX_CA_MAX 0x32
  27. enum {
  28. STATUS_PORT_STARTED, /* track if AFE port has started */
  29. STATUS_MAX
  30. };
  31. struct msm_ext_disp_ca {
  32. bool set_ca;
  33. u32 ca;
  34. };
  35. struct msm_dai_q6_hdmi_dai_data {
  36. DECLARE_BITMAP(status_mask, STATUS_MAX);
  37. u32 rate;
  38. u32 channels;
  39. u32 stream_idx;
  40. u32 ctl_idx;
  41. struct msm_ext_disp_ca ca;
  42. union afe_port_config port_config;
  43. };
  44. static int msm_dai_q6_ext_disp_format_put(struct snd_kcontrol *kcontrol,
  45. struct snd_ctl_elem_value *ucontrol)
  46. {
  47. struct msm_dai_q6_hdmi_dai_data *dai_data = kcontrol->private_data;
  48. int value = ucontrol->value.integer.value[0];
  49. if (!dai_data) {
  50. pr_err("%s: dai_data is NULL\n", __func__);
  51. return -EINVAL;
  52. }
  53. dai_data->port_config.hdmi_multi_ch.datatype = value;
  54. pr_debug("%s: value = %d\n", __func__, value);
  55. return 0;
  56. }
  57. static int msm_dai_q6_ext_disp_format_get(struct snd_kcontrol *kcontrol,
  58. struct snd_ctl_elem_value *ucontrol)
  59. {
  60. struct msm_dai_q6_hdmi_dai_data *dai_data = kcontrol->private_data;
  61. if (!dai_data) {
  62. pr_err("%s: dai_data is NULL\n", __func__);
  63. return -EINVAL;
  64. }
  65. ucontrol->value.integer.value[0] =
  66. dai_data->port_config.hdmi_multi_ch.datatype;
  67. pr_debug("%s: value = %ld\n",
  68. __func__, ucontrol->value.integer.value[0]);
  69. return 0;
  70. }
  71. static int msm_dai_q6_ext_disp_device_idx_put(struct snd_kcontrol *kcontrol,
  72. struct snd_ctl_elem_value *ucontrol)
  73. {
  74. struct msm_dai_q6_hdmi_dai_data *dai_data = kcontrol->private_data;
  75. if (!dai_data) {
  76. pr_err("%s: dai_data is NULL\n", __func__);
  77. return -EINVAL;
  78. }
  79. dai_data->ctl_idx = ucontrol->value.integer.value[0];
  80. dai_data->stream_idx = ucontrol->value.integer.value[1];
  81. pr_debug("%s: DP ctl id %d stream id %d\n", __func__,
  82. dai_data->ctl_idx, dai_data->stream_idx);
  83. return 0;
  84. }
  85. static int msm_dai_q6_ext_disp_device_idx_get(struct snd_kcontrol *kcontrol,
  86. struct snd_ctl_elem_value *ucontrol)
  87. {
  88. struct msm_dai_q6_hdmi_dai_data *dai_data = kcontrol->private_data;
  89. if (!dai_data) {
  90. pr_err("%s: dai_data is NULL\n", __func__);
  91. return -EINVAL;
  92. }
  93. ucontrol->value.integer.value[0] = dai_data->ctl_idx;
  94. ucontrol->value.integer.value[1] = dai_data->stream_idx;
  95. pr_debug("%s: DP ctl id %d stream id %d\n", __func__,
  96. dai_data->ctl_idx, dai_data->stream_idx);
  97. return 0;
  98. }
  99. static int msm_dai_q6_ext_disp_ca_put(struct snd_kcontrol *kcontrol,
  100. struct snd_ctl_elem_value *ucontrol)
  101. {
  102. struct msm_dai_q6_hdmi_dai_data *dai_data = kcontrol->private_data;
  103. if (!dai_data) {
  104. pr_err("%s: dai_data is NULL\n", __func__);
  105. return -EINVAL;
  106. }
  107. dai_data->ca.ca = ucontrol->value.integer.value[0];
  108. dai_data->ca.set_ca = true;
  109. pr_debug("%s: ca = %d\n", __func__, dai_data->ca.ca);
  110. return 0;
  111. }
  112. static int msm_dai_q6_ext_disp_ca_get(struct snd_kcontrol *kcontrol,
  113. struct snd_ctl_elem_value *ucontrol)
  114. {
  115. struct msm_dai_q6_hdmi_dai_data *dai_data = kcontrol->private_data;
  116. if (!dai_data) {
  117. pr_err("%s: dai_data is NULL\n", __func__);
  118. return -EINVAL;
  119. }
  120. ucontrol->value.integer.value[0] = dai_data->ca.ca;
  121. pr_debug("%s: ca = %d\n", __func__, dai_data->ca.ca);
  122. return 0;
  123. }
  124. /* HDMI format field for AFE_PORT_MULTI_CHAN_HDMI_AUDIO_IF_CONFIG command
  125. * 0: linear PCM
  126. * 1: non-linear PCM
  127. */
  128. static const char * const hdmi_format[] = {
  129. "LPCM",
  130. "Compr"
  131. };
  132. static const struct soc_enum hdmi_config_enum[] = {
  133. SOC_ENUM_SINGLE_EXT(2, hdmi_format),
  134. };
  135. static int msm_dai_q6_ext_disp_drift_info(struct snd_kcontrol *kcontrol,
  136. struct snd_ctl_elem_info *uinfo)
  137. {
  138. uinfo->type = SNDRV_CTL_ELEM_TYPE_BYTES;
  139. uinfo->count = sizeof(struct afe_param_id_dev_timing_stats);
  140. return 0;
  141. }
  142. static int msm_dai_q6_ext_disp_drift_get(struct snd_kcontrol *kcontrol,
  143. struct snd_ctl_elem_value *ucontrol)
  144. {
  145. int ret = -EINVAL;
  146. struct afe_param_id_dev_timing_stats timing_stats;
  147. struct snd_soc_dai *dai = kcontrol->private_data;
  148. struct msm_dai_q6_hdmi_dai_data *dai_data = dev_get_drvdata(dai->dev);
  149. if (!test_bit(STATUS_PORT_STARTED, dai_data->status_mask)) {
  150. pr_err("%s: afe port not started. status_mask = %ld\n",
  151. __func__, *dai_data->status_mask);
  152. goto done;
  153. }
  154. memset(&timing_stats, 0, sizeof(struct afe_param_id_dev_timing_stats));
  155. ret = afe_get_av_dev_drift(&timing_stats, dai->id);
  156. if (ret) {
  157. pr_err("%s: Error getting AFE Drift for port %d, err=%d\n",
  158. __func__, dai->id, ret);
  159. ret = -EINVAL;
  160. goto done;
  161. }
  162. memcpy(ucontrol->value.bytes.data, (void *)&timing_stats,
  163. sizeof(struct afe_param_id_dev_timing_stats));
  164. done:
  165. return ret;
  166. }
  167. static const struct snd_kcontrol_new hdmi_config_controls[] = {
  168. SOC_ENUM_EXT("HDMI RX Format", hdmi_config_enum[0],
  169. msm_dai_q6_ext_disp_format_get,
  170. msm_dai_q6_ext_disp_format_put),
  171. SOC_SINGLE_MULTI_EXT("HDMI RX CA", SND_SOC_NOPM, 0,
  172. HDMI_RX_CA_MAX, 0, 1,
  173. msm_dai_q6_ext_disp_ca_get,
  174. msm_dai_q6_ext_disp_ca_put),
  175. {
  176. .access = SNDRV_CTL_ELEM_ACCESS_READ,
  177. .iface = SNDRV_CTL_ELEM_IFACE_PCM,
  178. .name = "HDMI DRIFT",
  179. .info = msm_dai_q6_ext_disp_drift_info,
  180. .get = msm_dai_q6_ext_disp_drift_get,
  181. },
  182. };
  183. static const struct snd_kcontrol_new display_port_config_controls[] = {
  184. SOC_ENUM_EXT("Display Port RX Format", hdmi_config_enum[0],
  185. msm_dai_q6_ext_disp_format_get,
  186. msm_dai_q6_ext_disp_format_put),
  187. SOC_SINGLE_MULTI_EXT("Display Port RX CA", SND_SOC_NOPM, 0,
  188. HDMI_RX_CA_MAX, 0, 1,
  189. msm_dai_q6_ext_disp_ca_get,
  190. msm_dai_q6_ext_disp_ca_put),
  191. SOC_SINGLE_MULTI_EXT("Display Port RX DEVICE IDX", SND_SOC_NOPM, 0,
  192. 1, 0, 1,
  193. msm_dai_q6_ext_disp_device_idx_get,
  194. msm_dai_q6_ext_disp_device_idx_put),
  195. {
  196. .access = SNDRV_CTL_ELEM_ACCESS_READ,
  197. .iface = SNDRV_CTL_ELEM_IFACE_PCM,
  198. .name = "DISPLAY_PORT DRIFT",
  199. .info = msm_dai_q6_ext_disp_drift_info,
  200. .get = msm_dai_q6_ext_disp_drift_get,
  201. },
  202. };
  203. /* Current implementation assumes hw_param is called once
  204. * This may not be the case but what to do when ADM and AFE
  205. * port are already opened and parameter changes
  206. */
  207. static int msm_dai_q6_hdmi_hw_params(struct snd_pcm_substream *substream,
  208. struct snd_pcm_hw_params *params,
  209. struct snd_soc_dai *dai)
  210. {
  211. struct msm_dai_q6_hdmi_dai_data *dai_data = dev_get_drvdata(dai->dev);
  212. dai_data->channels = params_channels(params);
  213. dai_data->rate = params_rate(params);
  214. dai_data->port_config.hdmi_multi_ch.reserved = 0;
  215. dai_data->port_config.hdmi_multi_ch.hdmi_cfg_minor_version = 1;
  216. dai_data->port_config.hdmi_multi_ch.sample_rate = dai_data->rate;
  217. switch (params_format(params)) {
  218. case SNDRV_PCM_FORMAT_S16_LE:
  219. dai_data->port_config.hdmi_multi_ch.bit_width = 16;
  220. break;
  221. case SNDRV_PCM_FORMAT_S24_LE:
  222. case SNDRV_PCM_FORMAT_S24_3LE:
  223. dai_data->port_config.hdmi_multi_ch.bit_width = 24;
  224. break;
  225. }
  226. /*refer to HDMI spec CEA-861-E: Table 28 Audio InfoFrame Data Byte 4*/
  227. switch (dai_data->channels) {
  228. case 2:
  229. dai_data->port_config.hdmi_multi_ch.channel_allocation = 0;
  230. break;
  231. case 3:
  232. dai_data->port_config.hdmi_multi_ch.channel_allocation = 0x02;
  233. break;
  234. case 4:
  235. dai_data->port_config.hdmi_multi_ch.channel_allocation = 0x06;
  236. break;
  237. case 5:
  238. dai_data->port_config.hdmi_multi_ch.channel_allocation = 0x0A;
  239. break;
  240. case 6:
  241. dai_data->port_config.hdmi_multi_ch.channel_allocation = 0x0B;
  242. break;
  243. case 7:
  244. dai_data->port_config.hdmi_multi_ch.channel_allocation = 0x12;
  245. break;
  246. case 8:
  247. dai_data->port_config.hdmi_multi_ch.channel_allocation = 0x13;
  248. break;
  249. default:
  250. dev_err(dai->dev, "invalid Channels = %u\n",
  251. dai_data->channels);
  252. return -EINVAL;
  253. }
  254. dev_dbg(dai->dev, "%s() minor version: %u samplerate: %u bitwidth: %u\n"
  255. "num_ch = %u channel_allocation = %u datatype = %d\n", __func__,
  256. dai_data->port_config.hdmi_multi_ch.hdmi_cfg_minor_version,
  257. dai_data->port_config.hdmi_multi_ch.sample_rate,
  258. dai_data->port_config.hdmi_multi_ch.bit_width,
  259. dai_data->channels,
  260. dai_data->port_config.hdmi_multi_ch.channel_allocation,
  261. dai_data->port_config.hdmi_multi_ch.datatype);
  262. return 0;
  263. }
  264. static void msm_dai_q6_hdmi_shutdown(struct snd_pcm_substream *substream,
  265. struct snd_soc_dai *dai)
  266. {
  267. struct msm_dai_q6_hdmi_dai_data *dai_data = dev_get_drvdata(dai->dev);
  268. int rc = 0;
  269. if (!test_bit(STATUS_PORT_STARTED, dai_data->status_mask)) {
  270. pr_info("%s: afe port not started. dai_data->status_mask = %ld\n",
  271. __func__, *dai_data->status_mask);
  272. return;
  273. }
  274. rc = afe_close(dai->id); /* can block */
  275. if (rc < 0)
  276. dev_err(dai->dev, "fail to close AFE port\n");
  277. pr_debug("%s: dai_data->status_mask = %ld\n", __func__,
  278. *dai_data->status_mask);
  279. clear_bit(STATUS_PORT_STARTED, dai_data->status_mask);
  280. }
  281. static int msm_dai_q6_hdmi_prepare(struct snd_pcm_substream *substream,
  282. struct snd_soc_dai *dai)
  283. {
  284. struct msm_dai_q6_hdmi_dai_data *dai_data = dev_get_drvdata(dai->dev);
  285. int rc = 0;
  286. if (dai_data->ca.set_ca)
  287. dai_data->port_config.hdmi_multi_ch.channel_allocation =
  288. dai_data->ca.ca;
  289. if (!test_bit(STATUS_PORT_STARTED, dai_data->status_mask)) {
  290. rc = afe_set_display_stream(dai->id, dai_data->stream_idx,
  291. dai_data->ctl_idx);
  292. if (rc < 0) {
  293. dev_err(dai->dev, "fail to set AFE ctl, stream ID params %x\n",
  294. dai->id);
  295. if (rc != -EOPNOTSUPP) {
  296. dev_err(dai->dev, "not starting AFE port\n");
  297. goto err;
  298. }
  299. }
  300. rc = afe_port_start(dai->id, &dai_data->port_config,
  301. dai_data->rate);
  302. if (rc < 0)
  303. dev_err(dai->dev, "fail to open AFE port %x\n",
  304. dai->id);
  305. else
  306. set_bit(STATUS_PORT_STARTED,
  307. dai_data->status_mask);
  308. }
  309. err:
  310. return rc;
  311. }
  312. static inline void msm_dai_q6_hdmi_set_dai_id(struct snd_soc_dai *dai)
  313. {
  314. if (!dai->driver->id) {
  315. dev_warn(dai->dev, "DAI driver id is not set\n");
  316. return;
  317. }
  318. dai->id = dai->driver->id;
  319. }
  320. static int msm_dai_q6_hdmi_dai_probe(struct snd_soc_dai *dai)
  321. {
  322. struct msm_dai_q6_hdmi_dai_data *dai_data;
  323. const struct snd_kcontrol_new *kcontrol;
  324. int rc = 0;
  325. struct snd_soc_dapm_route intercon;
  326. struct snd_soc_dapm_context *dapm;
  327. if (!dai || !dai->driver) {
  328. pr_err("%s: dai or dai->driver is NULL\n", __func__);
  329. return -EINVAL;
  330. }
  331. dai_data = kzalloc(sizeof(struct msm_dai_q6_hdmi_dai_data),
  332. GFP_KERNEL);
  333. if (!dai_data) {
  334. dev_err(dai->dev, "DAI-%d: fail to allocate dai data\n",
  335. dai->id);
  336. rc = -ENOMEM;
  337. } else
  338. dev_set_drvdata(dai->dev, dai_data);
  339. msm_dai_q6_hdmi_set_dai_id(dai);
  340. if (dai->driver->id == HDMI_RX) {
  341. kcontrol = &hdmi_config_controls[0];
  342. rc = snd_ctl_add(dai->component->card->snd_card,
  343. snd_ctl_new1(kcontrol, dai_data));
  344. kcontrol = &hdmi_config_controls[1];
  345. rc = snd_ctl_add(dai->component->card->snd_card,
  346. snd_ctl_new1(kcontrol, dai_data));
  347. kcontrol = &hdmi_config_controls[2];
  348. rc = snd_ctl_add(dai->component->card->snd_card,
  349. snd_ctl_new1(kcontrol, dai));
  350. } else if (dai->driver->id == DISPLAY_PORT_RX) {
  351. kcontrol = &display_port_config_controls[0];
  352. rc = snd_ctl_add(dai->component->card->snd_card,
  353. snd_ctl_new1(kcontrol, dai_data));
  354. kcontrol = &display_port_config_controls[1];
  355. rc = snd_ctl_add(dai->component->card->snd_card,
  356. snd_ctl_new1(kcontrol, dai_data));
  357. kcontrol = &display_port_config_controls[2];
  358. rc = snd_ctl_add(dai->component->card->snd_card,
  359. snd_ctl_new1(kcontrol, dai_data));
  360. kcontrol = &display_port_config_controls[3];
  361. rc = snd_ctl_add(dai->component->card->snd_card,
  362. snd_ctl_new1(kcontrol, dai));
  363. } else {
  364. dev_err(dai->dev, "%s: Invalid id:%d\n",
  365. __func__, dai->driver->id);
  366. kfree(dai_data);
  367. dev_set_drvdata(dai->dev, NULL);
  368. return -EINVAL;
  369. }
  370. dapm = snd_soc_component_get_dapm(dai->component);
  371. memset(&intercon, 0, sizeof(intercon));
  372. if (!rc) {
  373. if (dai->driver->playback.stream_name &&
  374. dai->driver->playback.aif_name) {
  375. dev_dbg(dai->dev, "%s add route for widget %s",
  376. __func__, dai->driver->playback.stream_name);
  377. intercon.source = dai->driver->playback.aif_name;
  378. intercon.sink = dai->driver->playback.stream_name;
  379. dev_dbg(dai->dev, "%s src %s sink %s\n",
  380. __func__, intercon.source, intercon.sink);
  381. snd_soc_dapm_add_routes(dapm, &intercon, 1);
  382. }
  383. if (dai->driver->capture.stream_name &&
  384. dai->driver->capture.aif_name) {
  385. dev_dbg(dai->dev, "%s add route for widget %s",
  386. __func__, dai->driver->capture.stream_name);
  387. intercon.sink = dai->driver->capture.aif_name;
  388. intercon.source = dai->driver->capture.stream_name;
  389. dev_dbg(dai->dev, "%s src %s sink %s\n",
  390. __func__, intercon.source, intercon.sink);
  391. snd_soc_dapm_add_routes(dapm, &intercon, 1);
  392. }
  393. }
  394. return rc;
  395. }
  396. static int msm_dai_q6_hdmi_dai_remove(struct snd_soc_dai *dai)
  397. {
  398. struct msm_dai_q6_hdmi_dai_data *dai_data;
  399. int rc;
  400. dai_data = dev_get_drvdata(dai->dev);
  401. /* If AFE port is still up, close it */
  402. if (test_bit(STATUS_PORT_STARTED, dai_data->status_mask)) {
  403. rc = afe_close(dai->id); /* can block */
  404. if (rc < 0)
  405. dev_err(dai->dev, "fail to close AFE port\n");
  406. clear_bit(STATUS_PORT_STARTED, dai_data->status_mask);
  407. }
  408. kfree(dai_data);
  409. return 0;
  410. }
  411. static struct snd_soc_dai_ops msm_dai_q6_hdmi_ops = {
  412. .prepare = msm_dai_q6_hdmi_prepare,
  413. .hw_params = msm_dai_q6_hdmi_hw_params,
  414. .shutdown = msm_dai_q6_hdmi_shutdown,
  415. };
  416. static struct snd_soc_dai_driver msm_dai_q6_hdmi_hdmi_rx_dai = {
  417. .playback = {
  418. .stream_name = "HDMI Playback",
  419. .aif_name = "HDMI",
  420. .rates = SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 |
  421. SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 |
  422. SNDRV_PCM_RATE_96000 | SNDRV_PCM_RATE_176400 |
  423. SNDRV_PCM_RATE_192000,
  424. .formats = SNDRV_PCM_FMTBIT_S16_LE |
  425. SNDRV_PCM_FMTBIT_S24_LE |
  426. SNDRV_PCM_FMTBIT_S24_3LE,
  427. .channels_min = 2,
  428. .channels_max = 8,
  429. .rate_max = 192000,
  430. .rate_min = 48000,
  431. },
  432. .ops = &msm_dai_q6_hdmi_ops,
  433. .id = HDMI_RX,
  434. .probe = msm_dai_q6_hdmi_dai_probe,
  435. .remove = msm_dai_q6_hdmi_dai_remove,
  436. };
  437. static struct snd_soc_dai_driver msm_dai_q6_display_port_rx_dai[] = {
  438. {
  439. .playback = {
  440. .stream_name = "Display Port Playback",
  441. .aif_name = "DISPLAY_PORT",
  442. .rates = SNDRV_PCM_RATE_32000 | SNDRV_PCM_RATE_44100 |
  443. SNDRV_PCM_RATE_48000 | SNDRV_PCM_RATE_88200 |
  444. SNDRV_PCM_RATE_96000 | SNDRV_PCM_RATE_176400 |
  445. SNDRV_PCM_RATE_192000,
  446. .formats = SNDRV_PCM_FMTBIT_S16_LE |
  447. SNDRV_PCM_FMTBIT_S24_LE |
  448. SNDRV_PCM_FMTBIT_S24_3LE,
  449. .channels_min = 2,
  450. .channels_max = 8,
  451. .rate_max = 192000,
  452. .rate_min = 48000,
  453. },
  454. .ops = &msm_dai_q6_hdmi_ops,
  455. .id = DISPLAY_PORT_RX,
  456. .probe = msm_dai_q6_hdmi_dai_probe,
  457. .remove = msm_dai_q6_hdmi_dai_remove,
  458. },
  459. };
  460. static const struct snd_soc_component_driver msm_dai_hdmi_q6_component = {
  461. .name = "msm-dai-q6-hdmi",
  462. };
  463. /* To do: change to register DAIs as batch */
  464. static int msm_dai_q6_hdmi_dev_probe(struct platform_device *pdev)
  465. {
  466. int rc, id;
  467. const char *q6_dev_id = "qcom,msm-dai-q6-dev-id";
  468. rc = of_property_read_u32(pdev->dev.of_node, q6_dev_id, &id);
  469. if (rc) {
  470. dev_err(&pdev->dev,
  471. "%s: missing %s in dt node\n", __func__, q6_dev_id);
  472. return rc;
  473. }
  474. pdev->id = id;
  475. pr_debug("%s: dev name %s, id:%d\n", __func__,
  476. dev_name(&pdev->dev), pdev->id);
  477. switch (pdev->id) {
  478. case HDMI_RX:
  479. rc = snd_soc_register_component(&pdev->dev,
  480. &msm_dai_hdmi_q6_component,
  481. &msm_dai_q6_hdmi_hdmi_rx_dai, 1);
  482. break;
  483. case DISPLAY_PORT_RX:
  484. rc = snd_soc_register_component(&pdev->dev,
  485. &msm_dai_hdmi_q6_component,
  486. &msm_dai_q6_display_port_rx_dai[0],
  487. ARRAY_SIZE(msm_dai_q6_display_port_rx_dai));
  488. break;
  489. default:
  490. dev_err(&pdev->dev, "invalid device ID %d\n", pdev->id);
  491. rc = -ENODEV;
  492. break;
  493. }
  494. return rc;
  495. }
  496. static int msm_dai_q6_hdmi_dev_remove(struct platform_device *pdev)
  497. {
  498. snd_soc_unregister_component(&pdev->dev);
  499. return 0;
  500. }
  501. static const struct of_device_id msm_dai_q6_hdmi_dt_match[] = {
  502. {.compatible = "qcom,msm-dai-q6-hdmi"},
  503. {}
  504. };
  505. MODULE_DEVICE_TABLE(of, msm_dai_q6_hdmi_dt_match);
  506. static struct platform_driver msm_dai_q6_hdmi_driver = {
  507. .probe = msm_dai_q6_hdmi_dev_probe,
  508. .remove = msm_dai_q6_hdmi_dev_remove,
  509. .driver = {
  510. .name = "msm-dai-q6-hdmi",
  511. .owner = THIS_MODULE,
  512. .of_match_table = msm_dai_q6_hdmi_dt_match,
  513. },
  514. };
  515. int __init msm_dai_q6_hdmi_init(void)
  516. {
  517. return platform_driver_register(&msm_dai_q6_hdmi_driver);
  518. }
  519. void msm_dai_q6_hdmi_exit(void)
  520. {
  521. platform_driver_unregister(&msm_dai_q6_hdmi_driver);
  522. }
  523. /* Module information */
  524. MODULE_DESCRIPTION("MSM DSP HDMI DAI driver");
  525. MODULE_LICENSE("GPL v2");