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

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