dp_audio.c 21 KB

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