dp_audio.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2022-2023, Qualcomm Innovation Center, Inc. All rights reserved.
  4. * Copyright (c) 2016-2021, The Linux Foundation. All rights reserved.
  5. */
  6. #include <linux/of_platform.h>
  7. #include <linux/soc/qcom/msm_ext_display.h>
  8. #include <linux/version.h>
  9. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 19, 0))
  10. #include <drm/display/drm_dp_helper.h>
  11. #else
  12. #include <drm/drm_dp_helper.h>
  13. #endif
  14. #include "dp_catalog.h"
  15. #include "dp_audio.h"
  16. #include "dp_panel.h"
  17. #include "dp_debug.h"
  18. #if defined(CONFIG_SECDP)
  19. #include "secdp.h"
  20. #if defined(CONFIG_SECDP_BIGDATA)
  21. #include <linux/secdp_bigdata.h>
  22. #endif
  23. #if defined(CONFIG_SECDP_SWITCH)
  24. #include <linux/switch.h>
  25. static struct switch_dev switch_secdp_audio = {
  26. .name = "ch_hdmi_audio",
  27. };
  28. #endif
  29. #endif/*CONFIG_SECDP*/
  30. struct dp_audio_private {
  31. struct platform_device *ext_pdev;
  32. struct platform_device *pdev;
  33. struct dp_catalog_audio *catalog;
  34. struct msm_ext_disp_init_data ext_audio_data;
  35. struct dp_panel *panel;
  36. bool ack_enabled;
  37. atomic_t session_on;
  38. bool engine_on;
  39. u32 channels;
  40. struct completion hpd_comp;
  41. struct workqueue_struct *notify_workqueue;
  42. struct delayed_work notify_delayed_work;
  43. struct mutex ops_lock;
  44. struct dp_audio dp_audio;
  45. atomic_t acked;
  46. };
  47. static u32 dp_audio_get_header(struct dp_catalog_audio *catalog,
  48. enum dp_catalog_audio_sdp_type sdp,
  49. enum dp_catalog_audio_header_type header)
  50. {
  51. catalog->sdp_type = sdp;
  52. catalog->sdp_header = header;
  53. catalog->get_header(catalog);
  54. return catalog->data;
  55. }
  56. static void dp_audio_set_header(struct dp_catalog_audio *catalog,
  57. u32 data,
  58. enum dp_catalog_audio_sdp_type sdp,
  59. enum dp_catalog_audio_header_type header)
  60. {
  61. catalog->sdp_type = sdp;
  62. catalog->sdp_header = header;
  63. catalog->data = data;
  64. catalog->set_header(catalog);
  65. }
  66. static void dp_audio_stream_sdp(struct dp_audio_private *audio)
  67. {
  68. struct dp_catalog_audio *catalog = audio->catalog;
  69. u32 value, new_value;
  70. u8 parity_byte;
  71. /* Config header and parity byte 1 */
  72. value = dp_audio_get_header(catalog,
  73. DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_1);
  74. value &= 0x0000ffff;
  75. new_value = 0x02;
  76. parity_byte = dp_header_get_parity(new_value);
  77. value |= ((new_value << HEADER_BYTE_1_BIT)
  78. | (parity_byte << PARITY_BYTE_1_BIT));
  79. DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
  80. value, parity_byte);
  81. dp_audio_set_header(catalog, value,
  82. DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_1);
  83. /* Config header and parity byte 2 */
  84. value = dp_audio_get_header(catalog,
  85. DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_2);
  86. value &= 0xffff0000;
  87. new_value = 0x0;
  88. parity_byte = dp_header_get_parity(new_value);
  89. value |= ((new_value << HEADER_BYTE_2_BIT)
  90. | (parity_byte << PARITY_BYTE_2_BIT));
  91. DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
  92. value, parity_byte);
  93. dp_audio_set_header(catalog, value,
  94. DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_2);
  95. /* Config header and parity byte 3 */
  96. value = dp_audio_get_header(catalog,
  97. DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_3);
  98. value &= 0x0000ffff;
  99. new_value = audio->channels - 1;
  100. parity_byte = dp_header_get_parity(new_value);
  101. value |= ((new_value << HEADER_BYTE_3_BIT)
  102. | (parity_byte << PARITY_BYTE_3_BIT));
  103. DP_DEBUG("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
  104. value, parity_byte);
  105. dp_audio_set_header(catalog, value,
  106. DP_AUDIO_SDP_STREAM, DP_AUDIO_SDP_HEADER_3);
  107. }
  108. static void dp_audio_timestamp_sdp(struct dp_audio_private *audio)
  109. {
  110. struct dp_catalog_audio *catalog = audio->catalog;
  111. u32 value, new_value;
  112. u8 parity_byte;
  113. /* Config header and parity byte 1 */
  114. value = dp_audio_get_header(catalog,
  115. DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_1);
  116. value &= 0x0000ffff;
  117. new_value = 0x1;
  118. parity_byte = dp_header_get_parity(new_value);
  119. value |= ((new_value << HEADER_BYTE_1_BIT)
  120. | (parity_byte << PARITY_BYTE_1_BIT));
  121. DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
  122. value, parity_byte);
  123. dp_audio_set_header(catalog, value,
  124. DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_1);
  125. /* Config header and parity byte 2 */
  126. value = dp_audio_get_header(catalog,
  127. DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_2);
  128. value &= 0xffff0000;
  129. new_value = 0x17;
  130. parity_byte = dp_header_get_parity(new_value);
  131. value |= ((new_value << HEADER_BYTE_2_BIT)
  132. | (parity_byte << PARITY_BYTE_2_BIT));
  133. DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
  134. value, parity_byte);
  135. dp_audio_set_header(catalog, value,
  136. DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_2);
  137. /* Config header and parity byte 3 */
  138. value = dp_audio_get_header(catalog,
  139. DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_3);
  140. value &= 0x0000ffff;
  141. new_value = (0x0 | (0x11 << 2));
  142. parity_byte = dp_header_get_parity(new_value);
  143. value |= ((new_value << HEADER_BYTE_3_BIT)
  144. | (parity_byte << PARITY_BYTE_3_BIT));
  145. DP_DEBUG("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
  146. value, parity_byte);
  147. dp_audio_set_header(catalog, value,
  148. DP_AUDIO_SDP_TIMESTAMP, DP_AUDIO_SDP_HEADER_3);
  149. }
  150. static void dp_audio_infoframe_sdp(struct dp_audio_private *audio)
  151. {
  152. struct dp_catalog_audio *catalog = audio->catalog;
  153. u32 value, new_value;
  154. u8 parity_byte;
  155. /* Config header and parity byte 1 */
  156. value = dp_audio_get_header(catalog,
  157. DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_1);
  158. value &= 0x0000ffff;
  159. new_value = 0x84;
  160. parity_byte = dp_header_get_parity(new_value);
  161. value |= ((new_value << HEADER_BYTE_1_BIT)
  162. | (parity_byte << PARITY_BYTE_1_BIT));
  163. DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
  164. value, parity_byte);
  165. dp_audio_set_header(catalog, value,
  166. DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_1);
  167. /* Config header and parity byte 2 */
  168. value = dp_audio_get_header(catalog,
  169. DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_2);
  170. value &= 0xffff0000;
  171. new_value = 0x1b;
  172. parity_byte = dp_header_get_parity(new_value);
  173. value |= ((new_value << HEADER_BYTE_2_BIT)
  174. | (parity_byte << PARITY_BYTE_2_BIT));
  175. DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
  176. value, parity_byte);
  177. dp_audio_set_header(catalog, value,
  178. DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_2);
  179. /* Config header and parity byte 3 */
  180. value = dp_audio_get_header(catalog,
  181. DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_3);
  182. value &= 0x0000ffff;
  183. new_value = (0x0 | (0x11 << 2));
  184. parity_byte = dp_header_get_parity(new_value);
  185. value |= ((new_value << HEADER_BYTE_3_BIT)
  186. | (parity_byte << PARITY_BYTE_3_BIT));
  187. DP_DEBUG("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
  188. new_value, parity_byte);
  189. dp_audio_set_header(catalog, value,
  190. DP_AUDIO_SDP_INFOFRAME, DP_AUDIO_SDP_HEADER_3);
  191. }
  192. static void dp_audio_copy_management_sdp(struct dp_audio_private *audio)
  193. {
  194. struct dp_catalog_audio *catalog = audio->catalog;
  195. u32 value, new_value;
  196. u8 parity_byte;
  197. /* Config header and parity byte 1 */
  198. value = dp_audio_get_header(catalog,
  199. DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_1);
  200. value &= 0x0000ffff;
  201. new_value = 0x05;
  202. parity_byte = dp_header_get_parity(new_value);
  203. value |= ((new_value << HEADER_BYTE_1_BIT)
  204. | (parity_byte << PARITY_BYTE_1_BIT));
  205. DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
  206. value, parity_byte);
  207. dp_audio_set_header(catalog, value,
  208. DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_1);
  209. /* Config header and parity byte 2 */
  210. value = dp_audio_get_header(catalog,
  211. DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_2);
  212. value &= 0xffff0000;
  213. new_value = 0x0F;
  214. parity_byte = dp_header_get_parity(new_value);
  215. value |= ((new_value << HEADER_BYTE_2_BIT)
  216. | (parity_byte << PARITY_BYTE_2_BIT));
  217. DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
  218. value, parity_byte);
  219. dp_audio_set_header(catalog, value,
  220. DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_2);
  221. /* Config header and parity byte 3 */
  222. value = dp_audio_get_header(catalog,
  223. DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_3);
  224. value &= 0x0000ffff;
  225. new_value = 0x0;
  226. parity_byte = dp_header_get_parity(new_value);
  227. value |= ((new_value << HEADER_BYTE_3_BIT)
  228. | (parity_byte << PARITY_BYTE_3_BIT));
  229. DP_DEBUG("Header Byte 3: value = 0x%x, parity_byte = 0x%x\n",
  230. value, parity_byte);
  231. dp_audio_set_header(catalog, value,
  232. DP_AUDIO_SDP_COPYMANAGEMENT, DP_AUDIO_SDP_HEADER_3);
  233. }
  234. static void dp_audio_isrc_sdp(struct dp_audio_private *audio)
  235. {
  236. struct dp_catalog_audio *catalog = audio->catalog;
  237. u32 value, new_value;
  238. u8 parity_byte;
  239. /* Config header and parity byte 1 */
  240. value = dp_audio_get_header(catalog,
  241. DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_1);
  242. value &= 0x0000ffff;
  243. new_value = 0x06;
  244. parity_byte = dp_header_get_parity(new_value);
  245. value |= ((new_value << HEADER_BYTE_1_BIT)
  246. | (parity_byte << PARITY_BYTE_1_BIT));
  247. DP_DEBUG("Header Byte 1: value = 0x%x, parity_byte = 0x%x\n",
  248. value, parity_byte);
  249. dp_audio_set_header(catalog, value,
  250. DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_1);
  251. /* Config header and parity byte 2 */
  252. value = dp_audio_get_header(catalog,
  253. DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_2);
  254. value &= 0xffff0000;
  255. new_value = 0x0F;
  256. parity_byte = dp_header_get_parity(new_value);
  257. value |= ((new_value << HEADER_BYTE_2_BIT)
  258. | (parity_byte << PARITY_BYTE_2_BIT));
  259. DP_DEBUG("Header Byte 2: value = 0x%x, parity_byte = 0x%x\n",
  260. value, parity_byte);
  261. dp_audio_set_header(catalog, value,
  262. DP_AUDIO_SDP_ISRC, DP_AUDIO_SDP_HEADER_2);
  263. }
  264. static void dp_audio_setup_sdp(struct dp_audio_private *audio)
  265. {
  266. if (!atomic_read(&audio->session_on)) {
  267. DP_WARN("session inactive\n");
  268. return;
  269. }
  270. /* always program stream 0 first before actual stream cfg */
  271. audio->catalog->stream_id = DP_STREAM_0;
  272. audio->catalog->config_sdp(audio->catalog);
  273. if (audio->panel->stream_id == DP_STREAM_1) {
  274. audio->catalog->stream_id = DP_STREAM_1;
  275. audio->catalog->config_sdp(audio->catalog);
  276. }
  277. dp_audio_stream_sdp(audio);
  278. dp_audio_timestamp_sdp(audio);
  279. dp_audio_infoframe_sdp(audio);
  280. dp_audio_copy_management_sdp(audio);
  281. dp_audio_isrc_sdp(audio);
  282. }
  283. static void dp_audio_setup_acr(struct dp_audio_private *audio)
  284. {
  285. u32 select = 0;
  286. struct dp_catalog_audio *catalog = audio->catalog;
  287. if (!atomic_read(&audio->session_on)) {
  288. DP_WARN("session inactive\n");
  289. return;
  290. }
  291. switch (audio->dp_audio.bw_code) {
  292. case DP_LINK_BW_1_62:
  293. select = 0;
  294. break;
  295. case DP_LINK_BW_2_7:
  296. select = 1;
  297. break;
  298. case DP_LINK_BW_5_4:
  299. select = 2;
  300. break;
  301. case DP_LINK_BW_8_1:
  302. select = 3;
  303. break;
  304. default:
  305. DP_DEBUG("Unknown link rate\n");
  306. select = 0;
  307. break;
  308. }
  309. catalog->data = select;
  310. catalog->config_acr(catalog);
  311. }
  312. static void dp_audio_enable(struct dp_audio_private *audio, bool enable)
  313. {
  314. struct dp_catalog_audio *catalog = audio->catalog;
  315. audio->engine_on = enable;
  316. if (!atomic_read(&audio->session_on)) {
  317. DP_WARN("session inactive. enable=%d\n", enable);
  318. return;
  319. }
  320. catalog->data = enable;
  321. if (audio->panel->get_panel_on(audio->panel))
  322. catalog->enable(catalog);
  323. }
  324. static struct dp_audio_private *dp_audio_get_data(struct platform_device *pdev)
  325. {
  326. struct msm_ext_disp_data *ext_data;
  327. struct dp_audio *dp_audio;
  328. if (!pdev) {
  329. DP_ERR("invalid input\n");
  330. return ERR_PTR(-ENODEV);
  331. }
  332. ext_data = platform_get_drvdata(pdev);
  333. if (!ext_data) {
  334. DP_ERR("invalid ext disp data\n");
  335. return ERR_PTR(-EINVAL);
  336. }
  337. dp_audio = ext_data->intf_data;
  338. if (!dp_audio) {
  339. DP_ERR("invalid intf data\n");
  340. return ERR_PTR(-EINVAL);
  341. }
  342. return container_of(dp_audio, struct dp_audio_private, dp_audio);
  343. }
  344. static int dp_audio_info_setup(struct platform_device *pdev,
  345. struct msm_ext_disp_audio_setup_params *params)
  346. {
  347. int rc = 0;
  348. struct dp_audio_private *audio;
  349. audio = dp_audio_get_data(pdev);
  350. if (IS_ERR(audio)) {
  351. rc = PTR_ERR(audio);
  352. return rc;
  353. }
  354. if (audio->dp_audio.tui_active) {
  355. DP_DEBUG("TUI session active\n");
  356. return 0;
  357. }
  358. #if defined(CONFIG_SECDP_BIGDATA)
  359. secdp_bigdata_save_item(BD_AUD_CH, params->num_of_channels);
  360. secdp_bigdata_save_item(BD_AUD_FREQ, params->sample_rate_hz);
  361. #endif
  362. mutex_lock(&audio->ops_lock);
  363. audio->channels = params->num_of_channels;
  364. if (audio->panel->stream_id >= DP_STREAM_MAX) {
  365. DP_ERR("invalid stream id: %d\n",
  366. audio->panel->stream_id);
  367. rc = -EINVAL;
  368. mutex_unlock(&audio->ops_lock);
  369. return rc;
  370. }
  371. dp_audio_setup_sdp(audio);
  372. dp_audio_setup_acr(audio);
  373. dp_audio_enable(audio, true);
  374. mutex_unlock(&audio->ops_lock);
  375. DP_DEBUG("audio stream configured\n");
  376. return rc;
  377. }
  378. #if defined(CONFIG_SECDP)
  379. static void secdp_audio_use_one_sampling_freq(u8 *adb, int size)
  380. {
  381. const int one_adb_size = 3;
  382. int adb_count;
  383. if (size <= 0)
  384. return;
  385. adb_count = size / one_adb_size;
  386. while (adb_count > 0) {
  387. if (adb[1] & BIT(2))
  388. adb[1] = BIT(2); /* 48kHz */
  389. else if (adb[1] & BIT(1))
  390. adb[1] = BIT(1); /* 44.1kHz */
  391. else if (adb[1] & BIT(0))
  392. adb[1] = BIT(0); /* 32kHz */
  393. else if (adb[1] & BIT(3))
  394. adb[1] = BIT(3); /* 88kHz */
  395. else if (adb[1] & BIT(4))
  396. adb[1] = BIT(4); /* 96kHz */
  397. else if (adb[1] & BIT(5))
  398. adb[1] = BIT(5); /* 176kHz */
  399. else if (adb[1] & BIT(6))
  400. adb[1] = BIT(6); /* 192kHz */
  401. adb += one_adb_size;
  402. adb_count--;
  403. }
  404. }
  405. #endif
  406. static int dp_audio_get_edid_blk(struct platform_device *pdev,
  407. struct msm_ext_disp_audio_edid_blk *blk)
  408. {
  409. int rc = 0;
  410. struct dp_audio_private *audio;
  411. struct sde_edid_ctrl *edid;
  412. if (!blk) {
  413. DP_ERR("invalid input\n");
  414. return -EINVAL;
  415. }
  416. audio = dp_audio_get_data(pdev);
  417. if (IS_ERR(audio)) {
  418. rc = PTR_ERR(audio);
  419. goto end;
  420. }
  421. if (!audio->panel || !audio->panel->edid_ctrl) {
  422. DP_ERR("invalid panel data\n");
  423. rc = -EINVAL;
  424. goto end;
  425. }
  426. edid = audio->panel->edid_ctrl;
  427. #if defined(CONFIG_SECDP)
  428. if (secdp_adapter_is_legacy())
  429. secdp_audio_use_one_sampling_freq(edid->audio_data_block, edid->adb_size);
  430. #endif
  431. blk->audio_data_blk = edid->audio_data_block;
  432. blk->audio_data_blk_size = edid->adb_size;
  433. #if defined(CONFIG_SECDP)
  434. print_hex_dump(KERN_DEBUG, "AUDIO_BLK: ",
  435. DUMP_PREFIX_NONE, 16, 1, blk->audio_data_blk,
  436. blk->audio_data_blk_size, false);
  437. secdp_logger_hex_dump(blk->audio_data_blk, "AUDIO_BLK:",
  438. blk->audio_data_blk_size);
  439. #endif
  440. blk->spk_alloc_data_blk = edid->spkr_alloc_data_block;
  441. blk->spk_alloc_data_blk_size = edid->sadb_size;
  442. end:
  443. return rc;
  444. }
  445. static int dp_audio_get_cable_status(struct platform_device *pdev, u32 vote)
  446. {
  447. int rc = 0;
  448. struct dp_audio_private *audio;
  449. audio = dp_audio_get_data(pdev);
  450. if (IS_ERR(audio)) {
  451. rc = PTR_ERR(audio);
  452. goto end;
  453. }
  454. return atomic_read(&audio->session_on);
  455. end:
  456. return rc;
  457. }
  458. static int dp_audio_get_intf_id(struct platform_device *pdev)
  459. {
  460. int rc = 0;
  461. struct dp_audio_private *audio;
  462. audio = dp_audio_get_data(pdev);
  463. if (IS_ERR(audio)) {
  464. rc = PTR_ERR(audio);
  465. goto end;
  466. }
  467. return EXT_DISPLAY_TYPE_DP;
  468. end:
  469. return rc;
  470. }
  471. static void dp_audio_teardown_done(struct platform_device *pdev)
  472. {
  473. struct dp_audio_private *audio;
  474. audio = dp_audio_get_data(pdev);
  475. if (IS_ERR(audio))
  476. return;
  477. if (audio->dp_audio.tui_active) {
  478. DP_DEBUG("TUI session active\n");
  479. return;
  480. }
  481. if (audio->panel->stream_id >= DP_STREAM_MAX) {
  482. DP_WARN("invalid stream id: %d\n",
  483. audio->panel->stream_id);
  484. return;
  485. }
  486. mutex_lock(&audio->ops_lock);
  487. dp_audio_enable(audio, false);
  488. mutex_unlock(&audio->ops_lock);
  489. atomic_set(&audio->acked, 1);
  490. complete_all(&audio->hpd_comp);
  491. DP_DEBUG("audio engine disabled\n");
  492. }
  493. static int dp_audio_ack_done(struct platform_device *pdev, u32 ack)
  494. {
  495. int rc = 0, ack_hpd;
  496. struct dp_audio_private *audio;
  497. audio = dp_audio_get_data(pdev);
  498. if (IS_ERR(audio)) {
  499. rc = PTR_ERR(audio);
  500. goto end;
  501. }
  502. if (ack & AUDIO_ACK_SET_ENABLE) {
  503. audio->ack_enabled = ack & AUDIO_ACK_ENABLE ?
  504. true : false;
  505. DP_DEBUG("audio ack feature %s\n",
  506. audio->ack_enabled ? "enabled" : "disabled");
  507. goto end;
  508. }
  509. if (!audio->ack_enabled)
  510. goto end;
  511. ack_hpd = ack & AUDIO_ACK_CONNECT;
  512. DP_DEBUG("acknowledging audio (%d)\n", ack_hpd);
  513. if (!audio->engine_on) {
  514. atomic_set(&audio->acked, 1);
  515. complete_all(&audio->hpd_comp);
  516. }
  517. end:
  518. return rc;
  519. }
  520. static int dp_audio_codec_ready(struct platform_device *pdev)
  521. {
  522. int rc = 0;
  523. struct dp_audio_private *audio;
  524. audio = dp_audio_get_data(pdev);
  525. if (IS_ERR(audio)) {
  526. DP_ERR("invalid input\n");
  527. rc = PTR_ERR(audio);
  528. goto end;
  529. }
  530. queue_delayed_work(audio->notify_workqueue,
  531. &audio->notify_delayed_work, HZ/4);
  532. end:
  533. return rc;
  534. }
  535. static int dp_audio_register_ext_disp(struct dp_audio_private *audio)
  536. {
  537. int rc = 0;
  538. struct device_node *pd = NULL;
  539. const char *phandle = "qcom,ext-disp";
  540. struct msm_ext_disp_init_data *ext;
  541. struct msm_ext_disp_audio_codec_ops *ops;
  542. ext = &audio->ext_audio_data;
  543. ops = &ext->codec_ops;
  544. ext->codec.type = EXT_DISPLAY_TYPE_DP;
  545. ext->codec.ctrl_id = 0;
  546. ext->codec.stream_id = audio->panel->stream_id;
  547. ext->pdev = audio->pdev;
  548. ext->intf_data = &audio->dp_audio;
  549. ops->audio_info_setup = dp_audio_info_setup;
  550. ops->get_audio_edid_blk = dp_audio_get_edid_blk;
  551. ops->cable_status = dp_audio_get_cable_status;
  552. ops->get_intf_id = dp_audio_get_intf_id;
  553. ops->teardown_done = dp_audio_teardown_done;
  554. ops->acknowledge = dp_audio_ack_done;
  555. ops->ready = dp_audio_codec_ready;
  556. if (!audio->pdev->dev.of_node) {
  557. DP_ERR("cannot find audio dev.of_node\n");
  558. rc = -ENODEV;
  559. goto end;
  560. }
  561. pd = of_parse_phandle(audio->pdev->dev.of_node, phandle, 0);
  562. if (!pd) {
  563. DP_ERR("cannot parse %s handle\n", phandle);
  564. rc = -ENODEV;
  565. goto end;
  566. }
  567. audio->ext_pdev = of_find_device_by_node(pd);
  568. if (!audio->ext_pdev) {
  569. DP_ERR("cannot find %s pdev\n", phandle);
  570. rc = -ENODEV;
  571. goto end;
  572. }
  573. #if IS_ENABLED(CONFIG_MSM_EXT_DISPLAY)
  574. rc = msm_ext_disp_register_intf(audio->ext_pdev, ext);
  575. if (rc)
  576. DP_ERR("failed to register disp\n");
  577. #endif
  578. end:
  579. if (pd)
  580. of_node_put(pd);
  581. return rc;
  582. }
  583. static int dp_audio_deregister_ext_disp(struct dp_audio_private *audio)
  584. {
  585. int rc = 0;
  586. struct device_node *pd = NULL;
  587. const char *phandle = "qcom,ext-disp";
  588. struct msm_ext_disp_init_data *ext;
  589. ext = &audio->ext_audio_data;
  590. if (!audio->pdev->dev.of_node) {
  591. DP_ERR("cannot find audio dev.of_node\n");
  592. rc = -ENODEV;
  593. goto end;
  594. }
  595. pd = of_parse_phandle(audio->pdev->dev.of_node, phandle, 0);
  596. if (!pd) {
  597. DP_ERR("cannot parse %s handle\n", phandle);
  598. rc = -ENODEV;
  599. goto end;
  600. }
  601. audio->ext_pdev = of_find_device_by_node(pd);
  602. if (!audio->ext_pdev) {
  603. DP_ERR("cannot find %s pdev\n", phandle);
  604. rc = -ENODEV;
  605. goto end;
  606. }
  607. #if IS_ENABLED(CONFIG_MSM_EXT_DISPLAY)
  608. rc = msm_ext_disp_deregister_intf(audio->ext_pdev, ext);
  609. if (rc)
  610. DP_ERR("failed to deregister disp\n");
  611. #endif
  612. end:
  613. return rc;
  614. }
  615. #if defined(CONFIG_SECDP_SWITCH)
  616. extern int secdp_get_audio_ch(void);
  617. #endif
  618. static int dp_audio_notify(struct dp_audio_private *audio, u32 state)
  619. {
  620. int rc = 0;
  621. struct msm_ext_disp_init_data *ext = &audio->ext_audio_data;
  622. atomic_set(&audio->acked, 0);
  623. if (!ext->intf_ops.audio_notify) {
  624. DP_ERR("audio notify not defined\n");
  625. goto end;
  626. }
  627. reinit_completion(&audio->hpd_comp);
  628. rc = ext->intf_ops.audio_notify(audio->ext_pdev,
  629. &ext->codec, state);
  630. if (rc)
  631. goto end;
  632. #if defined(CONFIG_SECDP_SWITCH)
  633. {
  634. if (!audio->dp_audio.has_mst) {
  635. int audio_ch = state ? secdp_get_audio_ch() : -1;
  636. switch_set_state(&switch_secdp_audio, audio_ch);
  637. DP_INFO("secdp audio state:0x%x\n", audio_ch);
  638. }
  639. }
  640. #endif
  641. if (atomic_read(&audio->acked))
  642. goto end;
  643. if (state == EXT_DISPLAY_CABLE_DISCONNECT && !audio->engine_on)
  644. goto end;
  645. if (state == EXT_DISPLAY_CABLE_CONNECT)
  646. goto end;
  647. rc = wait_for_completion_timeout(&audio->hpd_comp, HZ * 4);
  648. if (!rc) {
  649. DP_ERR("timeout. state=%d err=%d\n", state, rc);
  650. rc = -ETIMEDOUT;
  651. goto end;
  652. }
  653. DP_DEBUG("success\n");
  654. end:
  655. return rc;
  656. }
  657. static int dp_audio_config(struct dp_audio_private *audio, u32 state)
  658. {
  659. int rc = 0;
  660. struct msm_ext_disp_init_data *ext = &audio->ext_audio_data;
  661. if (!ext || !ext->intf_ops.audio_config) {
  662. DP_ERR("audio_config not defined\n");
  663. goto end;
  664. }
  665. /*
  666. * DP Audio sets default STREAM_0 only, other streams are
  667. * set by audio driver based on the hardware/software support.
  668. */
  669. if (audio->panel->stream_id == DP_STREAM_0) {
  670. rc = ext->intf_ops.audio_config(audio->ext_pdev,
  671. &ext->codec, state);
  672. if (rc)
  673. DP_ERR("failed to config audio, err=%d\n",
  674. rc);
  675. }
  676. end:
  677. return rc;
  678. }
  679. static int dp_audio_on(struct dp_audio *dp_audio)
  680. {
  681. int rc = 0;
  682. struct dp_audio_private *audio;
  683. struct msm_ext_disp_init_data *ext;
  684. if (!dp_audio) {
  685. DP_ERR("invalid input\n");
  686. return -EINVAL;
  687. }
  688. #if defined(CONFIG_SECDP)
  689. if (!secdp_get_cable_status()) {
  690. DP_INFO("cable is out\n");
  691. return -EINVAL;
  692. }
  693. #endif
  694. audio = container_of(dp_audio, struct dp_audio_private, dp_audio);
  695. if (IS_ERR(audio)) {
  696. DP_ERR("invalid input\n");
  697. return -EINVAL;
  698. }
  699. dp_audio_register_ext_disp(audio);
  700. ext = &audio->ext_audio_data;
  701. atomic_set(&audio->session_on, 1);
  702. rc = dp_audio_config(audio, EXT_DISPLAY_CABLE_CONNECT);
  703. if (rc)
  704. goto end;
  705. rc = dp_audio_notify(audio, EXT_DISPLAY_CABLE_CONNECT);
  706. if (rc)
  707. goto end;
  708. DP_DEBUG("success\n");
  709. end:
  710. return rc;
  711. }
  712. static int dp_audio_off(struct dp_audio *dp_audio, bool skip_wait)
  713. {
  714. int rc = 0;
  715. struct dp_audio_private *audio;
  716. struct msm_ext_disp_init_data *ext;
  717. bool work_pending = false;
  718. if (!dp_audio) {
  719. DP_ERR("invalid input\n");
  720. return -EINVAL;
  721. }
  722. audio = container_of(dp_audio, struct dp_audio_private, dp_audio);
  723. if (!atomic_read(&audio->session_on)) {
  724. DP_DEBUG("audio already off\n");
  725. return rc;
  726. }
  727. ext = &audio->ext_audio_data;
  728. #if defined(CONFIG_SECDP)
  729. if (!atomic_read(&audio->session_on)) {
  730. DP_INFO("dp audio already off\n");
  731. return rc;
  732. }
  733. #endif
  734. work_pending = cancel_delayed_work_sync(&audio->notify_delayed_work);
  735. if (work_pending)
  736. DP_DEBUG("pending notification work completed\n");
  737. if (!skip_wait) {
  738. rc = dp_audio_notify(audio, EXT_DISPLAY_CABLE_DISCONNECT);
  739. if (rc)
  740. goto end;
  741. }
  742. DP_DEBUG("success\n");
  743. end:
  744. dp_audio_config(audio, EXT_DISPLAY_CABLE_DISCONNECT);
  745. atomic_set(&audio->session_on, 0);
  746. audio->engine_on = false;
  747. dp_audio_deregister_ext_disp(audio);
  748. return rc;
  749. }
  750. static void dp_audio_notify_work_fn(struct work_struct *work)
  751. {
  752. struct dp_audio_private *audio;
  753. struct delayed_work *dw = to_delayed_work(work);
  754. audio = container_of(dw, struct dp_audio_private, notify_delayed_work);
  755. dp_audio_notify(audio, EXT_DISPLAY_CABLE_CONNECT);
  756. }
  757. #if defined(CONFIG_SECDP_SWITCH)
  758. int secdp_audio_register_switch(struct dp_audio *dp_audio)
  759. {
  760. int rc = 0;
  761. if (dp_audio->has_mst) {
  762. DP_INFO("skip switch register\n");
  763. rc = -ENODEV;
  764. goto end;
  765. }
  766. rc = switch_dev_register(&switch_secdp_audio);
  767. if (rc) {
  768. DP_INFO("Failed to register secdp_audio switch %d\n", rc);
  769. rc = -ENODEV;
  770. goto end;
  771. }
  772. DP_INFO("secdp_audio register success\n");
  773. end:
  774. return rc;
  775. }
  776. static void secdp_audio_unregister_switch(struct dp_audio_private *audio)
  777. {
  778. struct dp_audio *dp_audio = &audio->dp_audio;
  779. if (dp_audio->has_mst) {
  780. DP_DEBUG("skip switch unregister\n");
  781. return;
  782. }
  783. switch_dev_unregister(&switch_secdp_audio);
  784. DP_INFO("secdp_audio unregister success\n");
  785. }
  786. #endif
  787. static int dp_audio_create_notify_workqueue(struct dp_audio_private *audio)
  788. {
  789. audio->notify_workqueue = create_workqueue("sdm_dp_audio_notify");
  790. if (IS_ERR_OR_NULL(audio->notify_workqueue)) {
  791. DP_ERR("Error creating notify_workqueue\n");
  792. return -EPERM;
  793. }
  794. INIT_DELAYED_WORK(&audio->notify_delayed_work, dp_audio_notify_work_fn);
  795. return 0;
  796. }
  797. static void dp_audio_destroy_notify_workqueue(struct dp_audio_private *audio)
  798. {
  799. if (audio->notify_workqueue)
  800. destroy_workqueue(audio->notify_workqueue);
  801. #if defined(CONFIG_SECDP_SWITCH)
  802. secdp_audio_unregister_switch(audio);
  803. #endif
  804. }
  805. struct dp_audio *dp_audio_get(struct platform_device *pdev,
  806. struct dp_panel *panel,
  807. struct dp_catalog_audio *catalog)
  808. {
  809. int rc = 0;
  810. struct dp_audio_private *audio;
  811. struct dp_audio *dp_audio;
  812. if (!pdev || !panel || !catalog) {
  813. DP_ERR("invalid input\n");
  814. rc = -EINVAL;
  815. goto error;
  816. }
  817. audio = devm_kzalloc(&pdev->dev, sizeof(*audio), GFP_KERNEL);
  818. if (!audio) {
  819. rc = -ENOMEM;
  820. goto error;
  821. }
  822. rc = dp_audio_create_notify_workqueue(audio);
  823. if (rc)
  824. goto error_notify_workqueue;
  825. init_completion(&audio->hpd_comp);
  826. audio->pdev = pdev;
  827. audio->panel = panel;
  828. audio->catalog = catalog;
  829. atomic_set(&audio->acked, 0);
  830. dp_audio = &audio->dp_audio;
  831. mutex_init(&audio->ops_lock);
  832. dp_audio->on = dp_audio_on;
  833. dp_audio->off = dp_audio_off;
  834. catalog->init(catalog);
  835. return dp_audio;
  836. error_notify_workqueue:
  837. devm_kfree(&pdev->dev, audio);
  838. error:
  839. return ERR_PTR(rc);
  840. }
  841. void dp_audio_put(struct dp_audio *dp_audio)
  842. {
  843. struct dp_audio_private *audio;
  844. if (!dp_audio)
  845. return;
  846. audio = container_of(dp_audio, struct dp_audio_private, dp_audio);
  847. mutex_destroy(&audio->ops_lock);
  848. dp_audio_destroy_notify_workqueue(audio);
  849. devm_kfree(&audio->pdev->dev, audio);
  850. }