gvm_auto_spf_dummy.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /* Copyright (c) 2014-2021, The Linux Foundation. All rights reserved.
  3. * Copyright (c) 2022-2023 Qualcomm Innovation Center, Inc. All rights reserved.
  4. */
  5. #include <linux/clk.h>
  6. #include <linux/delay.h>
  7. #include <linux/gpio.h>
  8. #include <linux/of_gpio.h>
  9. #include <linux/platform_device.h>
  10. #include <linux/slab.h>
  11. #include <linux/io.h>
  12. #include <linux/module.h>
  13. #include <linux/input.h>
  14. #include <linux/of_device.h>
  15. #include <linux/pm_qos.h>
  16. #include <sound/core.h>
  17. #include <sound/soc.h>
  18. #include <sound/soc-dapm.h>
  19. #include <sound/pcm.h>
  20. #include <sound/pcm_params.h>
  21. #include <sound/info.h>
  22. #include <dsp/audio_notifier.h>
  23. #include "msm_dailink.h"
  24. #include <soc/qcom/boot_stats.h>
  25. #include "msm_common.h"
  26. #define DRV_NAME "spf-asoc-snd"
  27. #define __CHIPSET__ "SA8xx5 "
  28. #define MSM_DAILINK_NAME(name) (__CHIPSET__#name)
  29. #define DEV_NAME_STR_LEN 32
  30. #define SAMPLING_RATE_8KHZ 8000
  31. #define SAMPLING_RATE_11P025KHZ 11025
  32. #define SAMPLING_RATE_16KHZ 16000
  33. #define SAMPLING_RATE_22P05KHZ 22050
  34. #define SAMPLING_RATE_32KHZ 32000
  35. #define SAMPLING_RATE_44P1KHZ 44100
  36. #define SAMPLING_RATE_48KHZ 48000
  37. #define SAMPLING_RATE_88P2KHZ 88200
  38. #define SAMPLING_RATE_96KHZ 96000
  39. #define SAMPLING_RATE_176P4KHZ 176400
  40. #define SAMPLING_RATE_192KHZ 192000
  41. #define SAMPLING_RATE_352P8KHZ 352800
  42. #define SAMPLING_RATE_384KHZ 384000
  43. #define WCD9XXX_MBHC_DEF_RLOADS 5
  44. #define WSA8810_NAME_1 "wsa881x.20170211"
  45. #define WSA8810_NAME_2 "wsa881x.20170212"
  46. #define WCN_CDC_SLIM_RX_CH_MAX 2
  47. #define WCN_CDC_SLIM_TX_CH_MAX 3
  48. #define TDM_CHANNEL_MAX 8
  49. #define ADSP_STATE_READY_TIMEOUT_MS 3000
  50. #define MSM_LL_QOS_VALUE 300 /* time in us to ensure LPM doesn't go in C3/C4 */
  51. #define MSM_HIFI_ON 1
  52. #define DIR_SZ 10
  53. struct snd_card_pdata {
  54. struct kobject snd_card_kobj;
  55. int card_status;
  56. } *snd_card_pdata;
  57. static struct attribute card_state_attr = {
  58. .name = "card_state",
  59. .mode = 0666,
  60. };
  61. enum {
  62. SLIM_RX_0 = 0,
  63. SLIM_RX_1,
  64. SLIM_RX_2,
  65. SLIM_RX_3,
  66. SLIM_RX_4,
  67. SLIM_RX_5,
  68. SLIM_RX_6,
  69. SLIM_RX_7,
  70. SLIM_RX_MAX,
  71. };
  72. enum {
  73. SLIM_TX_0 = 0,
  74. SLIM_TX_1,
  75. SLIM_TX_2,
  76. SLIM_TX_3,
  77. SLIM_TX_4,
  78. SLIM_TX_5,
  79. SLIM_TX_6,
  80. SLIM_TX_7,
  81. SLIM_TX_8,
  82. SLIM_TX_MAX,
  83. };
  84. enum {
  85. PRIM_MI2S = 0,
  86. SEC_MI2S,
  87. TERT_MI2S,
  88. QUAT_MI2S,
  89. QUIN_MI2S,
  90. MI2S_MAX,
  91. };
  92. enum {
  93. PRIM_AUX_PCM = 0,
  94. SEC_AUX_PCM,
  95. TERT_AUX_PCM,
  96. QUAT_AUX_PCM,
  97. QUIN_AUX_PCM,
  98. AUX_PCM_MAX,
  99. };
  100. struct mi2s_conf {
  101. struct mutex lock;
  102. u32 ref_cnt;
  103. u32 msm_is_mi2s_master;
  104. };
  105. struct dev_config {
  106. u32 sample_rate;
  107. u32 bit_format;
  108. u32 channels;
  109. };
  110. enum {
  111. DP_RX_IDX = 0,
  112. EXT_DISP_RX_IDX_MAX,
  113. };
  114. struct msm_wsa881x_dev_info {
  115. struct device_node *of_node;
  116. u32 index;
  117. };
  118. enum pinctrl_pin_state {
  119. STATE_DISABLE = 0, /* All pins are in sleep state */
  120. STATE_MI2S_ACTIVE, /* IS2 = active, TDM = sleep */
  121. STATE_TDM_ACTIVE, /* IS2 = sleep, TDM = active */
  122. };
  123. struct msm_pinctrl_info {
  124. struct pinctrl *pinctrl;
  125. struct pinctrl_state *mi2s_disable;
  126. struct pinctrl_state *tdm_disable;
  127. struct pinctrl_state *mi2s_active;
  128. struct pinctrl_state *tdm_active;
  129. enum pinctrl_pin_state curr_state;
  130. };
  131. enum {
  132. TDM_0 = 0,
  133. TDM_1,
  134. TDM_2,
  135. TDM_3,
  136. TDM_4,
  137. TDM_5,
  138. TDM_6,
  139. TDM_7,
  140. TDM_PORT_MAX,
  141. };
  142. enum {
  143. TDM_PRI = 0,
  144. TDM_SEC,
  145. TDM_TERT,
  146. TDM_QUAT,
  147. TDM_QUIN,
  148. TDM_INTERFACE_MAX,
  149. };
  150. struct tdm_port {
  151. u32 mode;
  152. u32 channel;
  153. };
  154. struct tdm_conf {
  155. struct mutex lock;
  156. u32 ref_cnt;
  157. };
  158. struct msm_asoc_mach_data {
  159. struct snd_info_entry *codec_root;
  160. struct msm_common_pdata *common_pdata;
  161. struct msm_pinctrl_info pinctrl_info;
  162. struct device_node *us_euro_gpio_p; /* used by pinctrl API */
  163. struct device_node *hph_en1_gpio_p; /* used by pinctrl API */
  164. struct device_node *hph_en0_gpio_p; /* used by pinctrl API */
  165. struct device_node *fsa_handle;
  166. struct snd_soc_codec *codec;
  167. struct work_struct adsp_power_up_work;
  168. struct tdm_conf tdm_intf_conf[TDM_INTERFACE_MAX];
  169. };
  170. static const char *const pin_states[] = {"sleep", "i2s-active",
  171. "tdm-active"};
  172. static struct platform_device *spdev;
  173. static bool codec_reg_done;
  174. /*
  175. * Need to report LINEIN
  176. * if R/L channel impedance is larger than 5K ohm
  177. */
  178. static const struct snd_soc_dapm_widget msm_dapm_widgets[] = {
  179. SND_SOC_DAPM_MIC("Handset Mic", NULL),
  180. SND_SOC_DAPM_MIC("Analog Mic3", NULL),
  181. SND_SOC_DAPM_MIC("Analog Mic4", NULL),
  182. SND_SOC_DAPM_MIC("Digital Mic0", NULL),
  183. SND_SOC_DAPM_MIC("Digital Mic1", NULL),
  184. SND_SOC_DAPM_MIC("Digital Mic2", NULL),
  185. SND_SOC_DAPM_MIC("Digital Mic3", NULL),
  186. SND_SOC_DAPM_MIC("Digital Mic4", NULL),
  187. SND_SOC_DAPM_MIC("Digital Mic5", NULL),
  188. SND_SOC_DAPM_MIC("Digital Mic6", NULL),
  189. SND_SOC_DAPM_MIC("Digital Mic7", NULL),
  190. };
  191. static struct snd_soc_dai_link msm_gvm_auto_dai_links[] = {
  192. /* BackEnd DAI Links */
  193. {
  194. .name = "PRI_TDM_RX_0_DUMMY",
  195. .stream_name = "TDM-LPAIF-RX-PRIMARY",
  196. .dpcm_playback = 1,
  197. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  198. SND_SOC_DPCM_TRIGGER_POST},
  199. .ignore_suspend = 1,
  200. .ignore_pmdown_time = 1,
  201. SND_SOC_DAILINK_REG(primary_tdm_rx_0_dummy),
  202. },
  203. {
  204. .name = "PRI_TDM_TX_0_DUMMY",
  205. .stream_name = "TDM-LPAIF-TX-PRIMARY",
  206. .dpcm_capture = 1,
  207. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  208. SND_SOC_DPCM_TRIGGER_POST},
  209. .ignore_suspend = 1,
  210. .ignore_pmdown_time = 1,
  211. SND_SOC_DAILINK_REG(primary_tdm_tx_0_dummy),
  212. },
  213. {
  214. .name = "SEC_TDM_RX_0_DUMMY",
  215. .stream_name = "TDM-LPAIF-RX-SECONDARY",
  216. .dpcm_playback = 1,
  217. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  218. SND_SOC_DPCM_TRIGGER_POST},
  219. .ignore_suspend = 1,
  220. .ignore_pmdown_time = 1,
  221. SND_SOC_DAILINK_REG(secondary_tdm_rx_0_dummy),
  222. },
  223. {
  224. .name = "SEC_TDM_TX_0_DUMMY",
  225. .stream_name = "TDM-LPAIF-TX-SECONDARY",
  226. .dpcm_capture = 1,
  227. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  228. SND_SOC_DPCM_TRIGGER_POST},
  229. .ignore_suspend = 1,
  230. .ignore_pmdown_time = 1,
  231. SND_SOC_DAILINK_REG(secondary_tdm_tx_0_dummy),
  232. },
  233. {
  234. .name = "TERT_TDM_RX_0_DUMMY",
  235. .stream_name = "TDM-LPAIF-RX-TERTIARY",
  236. .dpcm_playback = 1,
  237. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  238. SND_SOC_DPCM_TRIGGER_POST},
  239. .ignore_suspend = 1,
  240. .ignore_pmdown_time = 1,
  241. SND_SOC_DAILINK_REG(tert_tdm_rx_0_dummy),
  242. },
  243. {
  244. .name = "TERT_TDM_TX_0_DUMMY",
  245. .stream_name = "TDM-LPAIF-TX-TERTIARY",
  246. .dpcm_capture = 1,
  247. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  248. SND_SOC_DPCM_TRIGGER_POST},
  249. .ignore_suspend = 1,
  250. .ignore_pmdown_time = 1,
  251. SND_SOC_DAILINK_REG(tert_tdm_tx_0_dummy),
  252. },
  253. {
  254. .name = "QUAT_TDM_RX_0_DUMMY",
  255. .stream_name = "TDM-LPAIF_RXTX-RX-PRIMARY",
  256. .dpcm_playback = 1,
  257. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  258. SND_SOC_DPCM_TRIGGER_POST},
  259. .ignore_suspend = 1,
  260. .ignore_pmdown_time = 1,
  261. SND_SOC_DAILINK_REG(quat_tdm_rx_0_dummy),
  262. },
  263. {
  264. .name = "QUAT_TDM_TX_0_DUMMY",
  265. .stream_name = "TDM-LPAIF_RXTX-TX-PRIMARY",
  266. .dpcm_capture = 1,
  267. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  268. SND_SOC_DPCM_TRIGGER_POST},
  269. .ignore_suspend = 1,
  270. .ignore_pmdown_time = 1,
  271. SND_SOC_DAILINK_REG(quat_tdm_tx_0_dummy),
  272. },
  273. {
  274. .name = "QUIN_TDM_RX_0_DUMMY",
  275. .stream_name = "TDM-LPAIF_VA-RX-PRIMARY",
  276. .dpcm_playback = 1,
  277. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  278. SND_SOC_DPCM_TRIGGER_POST},
  279. .ignore_suspend = 1,
  280. .ignore_pmdown_time = 1,
  281. SND_SOC_DAILINK_REG(quin_tdm_rx_0_dummy),
  282. },
  283. {
  284. .name = "QUIN_TDM_TX_0_DUMMY",
  285. .stream_name = "TDM-LPAIF_VA-TX-PRIMARY",
  286. .dpcm_capture = 1,
  287. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  288. SND_SOC_DPCM_TRIGGER_POST},
  289. .ignore_suspend = 1,
  290. .ignore_pmdown_time = 1,
  291. SND_SOC_DAILINK_REG(quin_tdm_tx_0_dummy),
  292. },
  293. {
  294. .name = "SEN_TDM_RX_0_DUMMY",
  295. .stream_name = "TDM-LPAIF_WSA-RX-PRIMARY",
  296. .dpcm_playback = 1,
  297. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  298. SND_SOC_DPCM_TRIGGER_POST},
  299. .ignore_suspend = 1,
  300. .ignore_pmdown_time = 1,
  301. SND_SOC_DAILINK_REG(sen_tdm_rx_0_dummy),
  302. },
  303. {
  304. .name = "SEN_TDM_TX_0_DUMMY",
  305. .stream_name = "TDM-LPAIF_WSA-TX-PRIMARY",
  306. .dpcm_capture = 1,
  307. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  308. SND_SOC_DPCM_TRIGGER_POST},
  309. .ignore_suspend = 1,
  310. .ignore_pmdown_time = 1,
  311. SND_SOC_DAILINK_REG(sen_tdm_tx_0_dummy),
  312. },
  313. {
  314. .name = "SEP_TDM_RX_0_DUMMY",
  315. .stream_name = "TDM-LPAIF_AUD-RX-PRIMARY",
  316. .dpcm_playback = 1,
  317. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  318. SND_SOC_DPCM_TRIGGER_POST},
  319. .ignore_suspend = 1,
  320. .ignore_pmdown_time = 1,
  321. SND_SOC_DAILINK_REG(sep_tdm_rx_0_dummy),
  322. },
  323. {
  324. .name = "SEP_TDM_TX_0_DUMMY",
  325. .stream_name = "TDM-LPAIF_AUD-TX-PRIMARY",
  326. .dpcm_capture = 1,
  327. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  328. SND_SOC_DPCM_TRIGGER_POST},
  329. .ignore_suspend = 1,
  330. .ignore_pmdown_time = 1,
  331. SND_SOC_DAILINK_REG(sep_tdm_tx_0_dummy),
  332. },
  333. {
  334. .name = "OCT_TDM_RX_0_DUMMY",
  335. .stream_name = "TDM-LPAIF_WSA2-RX-PRIMARY",
  336. .dpcm_playback = 1,
  337. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  338. SND_SOC_DPCM_TRIGGER_POST},
  339. .ignore_suspend = 1,
  340. .ignore_pmdown_time = 1,
  341. SND_SOC_DAILINK_REG(oct_tdm_rx_0_dummy),
  342. },
  343. {
  344. .name = "OCT_TDM_TX_0_DUMMY",
  345. .stream_name = "TDM-LPAIF_WSA2-TX-PRIMARY",
  346. .dpcm_capture = 1,
  347. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  348. SND_SOC_DPCM_TRIGGER_POST},
  349. .ignore_suspend = 1,
  350. .ignore_pmdown_time = 1,
  351. SND_SOC_DAILINK_REG(oct_tdm_tx_0_dummy),
  352. },
  353. {
  354. .name = "HS_IF0_TDM_RX_0_DUMMY",
  355. .stream_name = "TDM-LPAIF_SDR-RX-PRIMARY",
  356. .dpcm_playback = 1,
  357. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  358. SND_SOC_DPCM_TRIGGER_POST},
  359. .ignore_suspend = 1,
  360. .ignore_pmdown_time = 1,
  361. SND_SOC_DAILINK_REG(hs_if0_tdm_rx_0_dummy),
  362. },
  363. {
  364. .name = "HS_IF0_TDM_TX_0_DUMMY",
  365. .stream_name = "TDM-LPAIF_SDR-TX-PRIMARY",
  366. .dpcm_capture = 1,
  367. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  368. SND_SOC_DPCM_TRIGGER_POST},
  369. .ignore_suspend = 1,
  370. .ignore_pmdown_time = 1,
  371. SND_SOC_DAILINK_REG(hs_if0_tdm_tx_0_dummy),
  372. },
  373. {
  374. .name = "HS_IF1_TDM_RX_0_DUMMY",
  375. .stream_name = "TDM-LPAIF_SDR-RX-SECONDARY",
  376. .dpcm_playback = 1,
  377. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  378. SND_SOC_DPCM_TRIGGER_POST},
  379. .ignore_suspend = 1,
  380. .ignore_pmdown_time = 1,
  381. SND_SOC_DAILINK_REG(hs_if1_tdm_rx_0_dummy),
  382. },
  383. {
  384. .name = "HS_IF1_TDM_TX_0_DUMMY",
  385. .stream_name = "TDM-LPAIF_SDR-TX-SECONDARY",
  386. .dpcm_capture = 1,
  387. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  388. SND_SOC_DPCM_TRIGGER_POST},
  389. .ignore_suspend = 1,
  390. .ignore_pmdown_time = 1,
  391. SND_SOC_DAILINK_REG(hs_if1_tdm_tx_0_dummy),
  392. },
  393. {
  394. .name = "HS_IF2_TDM_RX_0_DUMMY",
  395. .stream_name = "TDM-LPAIF_SDR-RX-TERTIARY",
  396. .dpcm_playback = 1,
  397. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  398. SND_SOC_DPCM_TRIGGER_POST},
  399. .ignore_suspend = 1,
  400. .ignore_pmdown_time = 1,
  401. SND_SOC_DAILINK_REG(hs_if2_tdm_rx_0_dummy),
  402. },
  403. {
  404. .name = "HS_IF2_TDM_TX_0_DUMMY",
  405. .stream_name = "TDM-LPAIF_SDR-TX-TERTIARY",
  406. .dpcm_capture = 1,
  407. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  408. SND_SOC_DPCM_TRIGGER_POST},
  409. .ignore_suspend = 1,
  410. .ignore_pmdown_time = 1,
  411. SND_SOC_DAILINK_REG(hs_if2_tdm_tx_0_dummy),
  412. },
  413. {
  414. .name = "HS_IF3_TDM_RX_0",
  415. .stream_name = "TDM-LPAIF_SDR-RX-QUATERNARY",
  416. .dpcm_playback = 1,
  417. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  418. SND_SOC_DPCM_TRIGGER_POST},
  419. .ignore_suspend = 1,
  420. .ignore_pmdown_time = 1,
  421. SND_SOC_DAILINK_REG(hs_if3_tdm_rx_0_dummy),
  422. },
  423. {
  424. .name = "HS_IF3_TDM_TX_0",
  425. .stream_name = "TDM-LPAIF_SDR-TX-QUATERNARY",
  426. .dpcm_capture = 1,
  427. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  428. SND_SOC_DPCM_TRIGGER_POST},
  429. .ignore_suspend = 1,
  430. .ignore_pmdown_time = 1,
  431. SND_SOC_DAILINK_REG(hs_if3_tdm_tx_0_dummy),
  432. },
  433. {
  434. .name = "HS_IF4_TDM_RX_0",
  435. .stream_name = "TDM-LPAIF_SDR-RX-QUINARY",
  436. .dpcm_playback = 1,
  437. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  438. SND_SOC_DPCM_TRIGGER_POST},
  439. .ignore_suspend = 1,
  440. .ignore_pmdown_time = 1,
  441. SND_SOC_DAILINK_REG(hs_if4_tdm_rx_0_dummy),
  442. },
  443. {
  444. .name = "HS_IF4_TDM_TX_0",
  445. .stream_name = "TDM-LPAIF_SDR-TX-QUINARY",
  446. .dpcm_capture = 1,
  447. .trigger = {SND_SOC_DPCM_TRIGGER_POST,
  448. SND_SOC_DPCM_TRIGGER_POST},
  449. .ignore_suspend = 1,
  450. .ignore_pmdown_time = 1,
  451. SND_SOC_DAILINK_REG(hs_if4_tdm_tx_0_dummy),
  452. },
  453. };
  454. int snd_card_notify_user(snd_card_status_t card_status)
  455. {
  456. snd_card_pdata->card_status = card_status;
  457. sysfs_notify(&snd_card_pdata->snd_card_kobj, NULL, "card_state");
  458. return 0;
  459. }
  460. int snd_card_set_card_status(snd_card_status_t card_status)
  461. {
  462. snd_card_pdata->card_status = card_status;
  463. return 0;
  464. }
  465. static ssize_t snd_card_sysfs_show(struct kobject *kobj,
  466. struct attribute *attr, char *buf)
  467. {
  468. /* Max length of buf is PAGE_SIZE */
  469. return sysfs_emit(buf, "%d\n", snd_card_pdata->card_status);
  470. }
  471. static ssize_t snd_card_sysfs_store(struct kobject *kobj,
  472. struct attribute *attr, const char *buf, size_t count)
  473. {
  474. int result;
  475. result = kstrtoint (buf, 0, &snd_card_pdata->card_status);
  476. if (result)
  477. pr_err("%s: Failed to read buf result=%d\n", __func__, result);
  478. sysfs_notify(kobj, NULL, "card_state");
  479. return 0;
  480. }
  481. static const struct sysfs_ops snd_card_sysfs_ops = {
  482. .show = snd_card_sysfs_show,
  483. .store = snd_card_sysfs_store,
  484. };
  485. static void snd_card_sysfs_release(struct kobject *kobj)
  486. {
  487. kfree(snd_card_pdata); /* Free the memory */
  488. }
  489. static struct kobj_type snd_card_ktype = {
  490. .release = snd_card_sysfs_release,
  491. .sysfs_ops = &snd_card_sysfs_ops,
  492. };
  493. int snd_card_sysfs_init(void)
  494. {
  495. int ret = 0;
  496. char dir[DIR_SZ] = "snd_card";
  497. snd_card_pdata = kcalloc(1, sizeof(struct snd_card_pdata), GFP_KERNEL);
  498. if (!snd_card_pdata)
  499. return -ENOMEM;
  500. /* kernel_kobj is the kobject of /sys/kernel/ */
  501. ret = kobject_init_and_add(&snd_card_pdata->snd_card_kobj, &snd_card_ktype,
  502. kernel_kobj, dir);
  503. if (ret < 0) {
  504. pr_err("%s: Failed to init and add kobject %s, err = %d\n",
  505. __func__, dir, ret);
  506. goto fail;
  507. }
  508. ret = sysfs_create_file(&snd_card_pdata->snd_card_kobj, &card_state_attr);
  509. if (ret < 0) {
  510. pr_err("%s: Failed to add snd_card sysfs entry to %s\n",
  511. __func__, dir);
  512. goto fail;
  513. }
  514. return ret;
  515. fail:
  516. kobject_put(&snd_card_pdata->snd_card_kobj);
  517. return ret;
  518. }
  519. struct snd_soc_card snd_soc_card_gvm_auto_dummy_msm;
  520. static int msm_populate_dai_link_component_of_node(
  521. struct snd_soc_card *card)
  522. {
  523. int i, j, index, ret = 0;
  524. struct device *cdev = card->dev;
  525. struct snd_soc_dai_link *dai_link = card->dai_link;
  526. struct device_node *np;
  527. if (!cdev) {
  528. pr_err("%s: Sound card device memory NULL\n", __func__);
  529. return -ENODEV;
  530. }
  531. for (i = 0; i < card->num_links; i++) {
  532. if (dai_link[i].platforms->of_node && dai_link[i].cpus->of_node)
  533. continue;
  534. /* populate cpu_of_node for snd card dai links */
  535. if (dai_link[i].cpus->dai_name && !dai_link[i].cpus->of_node) {
  536. index = of_property_match_string(cdev->of_node,
  537. "asoc-cpu-names",
  538. dai_link[i].cpus->dai_name);
  539. pr_err("%s: retrieving cpu_of_node for %s\n",
  540. __func__,
  541. dai_link[i].cpus->dai_name);
  542. if (index >= 0) {
  543. np = of_parse_phandle(cdev->of_node, "asoc-cpu",
  544. index);
  545. if (!np) {
  546. pr_err("%s: retrieving phandle for cpu dai %s failed\n",
  547. __func__,
  548. dai_link[i].cpus->dai_name);
  549. ret = -ENODEV;
  550. goto err;
  551. }
  552. dai_link[i].cpus->of_node = np;
  553. dai_link[i].cpus->dai_name = NULL;
  554. }
  555. }
  556. /* populate codec_of_node for snd card dai links */
  557. if (dai_link[i].num_codecs > 0) {
  558. for (j = 0; j < dai_link[i].num_codecs; j++) {
  559. if (dai_link[i].codecs[j].of_node ||
  560. !dai_link[i].codecs[j].name)
  561. continue;
  562. index = of_property_match_string(cdev->of_node,
  563. "asoc-codec-names",
  564. dai_link[i].codecs[j].name);
  565. if (index < 0)
  566. continue;
  567. np = of_parse_phandle(cdev->of_node,
  568. "asoc-codec",
  569. index);
  570. if (!np) {
  571. pr_err("%s: retrieving phandle for codec %s failed\n",
  572. __func__, dai_link[i].codecs[j].name);
  573. ret = -ENODEV;
  574. goto err;
  575. }
  576. dai_link[i].codecs[j].of_node = np;
  577. dai_link[i].codecs[j].name = NULL;
  578. }
  579. }
  580. }
  581. err:
  582. return ret;
  583. }
  584. static const struct of_device_id gvm_asoc_machine_of_match[] = {
  585. { .compatible = "qcom,8155-spf-asoc-snd-adp-star",
  586. .data = "adp_star_codec"},
  587. { .compatible = "qcom,6155-spf-asoc-snd-adp-star",
  588. .data = "adp_star_codec"},
  589. { .compatible = "qcom,gvm-auto-spf-asoc-snd-adp-star",
  590. .data = "adp_star_codec"},
  591. {},
  592. };
  593. static struct snd_soc_card *populate_snd_card_dailinks(struct device *dev)
  594. {
  595. struct snd_soc_card *card = NULL;
  596. const struct of_device_id *match = NULL;
  597. match = of_match_node(gvm_asoc_machine_of_match, dev->of_node);
  598. if (!match) {
  599. dev_err(dev, "%s: No DT match found for sound card\n",
  600. __func__);
  601. return NULL;
  602. }
  603. card = &snd_soc_card_gvm_auto_dummy_msm;
  604. if (!strcmp(match->compatible, "qcom,gvm-auto-spf-asoc-snd-adp-star")) {
  605. card->dai_link = msm_gvm_auto_dai_links;
  606. card->num_links = ARRAY_SIZE(msm_gvm_auto_dai_links);
  607. }
  608. return card;
  609. }
  610. struct msm_common_pdata *msm_common_get_pdata(struct snd_soc_card *card)
  611. {
  612. struct msm_asoc_mach_data *pdata = snd_soc_card_get_drvdata(card);
  613. if (!pdata)
  614. return NULL;
  615. return pdata->common_pdata;
  616. }
  617. void msm_common_set_pdata(struct snd_soc_card *card,
  618. struct msm_common_pdata *common_pdata)
  619. {
  620. struct msm_asoc_mach_data *pdata = snd_soc_card_get_drvdata(card);
  621. if (!pdata)
  622. return;
  623. pdata->common_pdata = common_pdata;
  624. }
  625. static int msm_asoc_machine_probe(struct platform_device *pdev)
  626. {
  627. struct snd_soc_card *card;
  628. struct msm_asoc_mach_data *pdata;
  629. int ret;
  630. pr_debug("%s: DRIVER Audio Init\n", __func__);
  631. if (!pdev->dev.of_node) {
  632. dev_err(&pdev->dev, "No platform supplied from device tree\n");
  633. return -EINVAL;
  634. }
  635. pdata = devm_kzalloc(&pdev->dev,
  636. sizeof(struct msm_asoc_mach_data), GFP_KERNEL);
  637. if (!pdata)
  638. return -ENOMEM;
  639. card = populate_snd_card_dailinks(&pdev->dev);
  640. if (!card) {
  641. dev_err(&pdev->dev, "%s: Card uninitialized\n", __func__);
  642. ret = -EINVAL;
  643. return ret;
  644. }
  645. card->dev = &pdev->dev;
  646. platform_set_drvdata(pdev, card);
  647. snd_soc_card_set_drvdata(card, pdata);
  648. ret = snd_soc_of_parse_card_name(card, "qcom,model");
  649. if (ret) {
  650. dev_err(&pdev->dev, "parse card name failed, err:%d\n",
  651. ret);
  652. return ret;
  653. }
  654. ret = msm_populate_dai_link_component_of_node(card);
  655. if (ret) {
  656. ret = -EPROBE_DEFER;
  657. return ret;
  658. }
  659. ret = devm_snd_soc_register_card(&pdev->dev, card);
  660. if (ret == -EPROBE_DEFER) {
  661. if (codec_reg_done)
  662. ret = -EINVAL;
  663. return ret;
  664. } else if (ret) {
  665. dev_err(&pdev->dev, "snd_soc_register_card failed (%d)\n",
  666. ret);
  667. pr_err("snd_soc_register_card failed (%d)\n",
  668. ret);
  669. return ret;
  670. }
  671. dev_info(&pdev->dev, "Sound card %s registered\n", card->name);
  672. pr_err("Sound card %s registered\n", card->name);
  673. pr_debug("%s: DRIVER Audio Ready\n", __func__);
  674. spdev = pdev;
  675. ret = snd_card_sysfs_init();
  676. if (ret)
  677. pr_err("snd_card_sysfs_init fail, ret=%d\n", ret);
  678. ret = snd_card_set_card_status(SND_CARD_STATUS_ONLINE);
  679. if (ret)
  680. pr_err("snd_card_set_card_status fail, ret=%d\n", ret);
  681. return 0;
  682. }
  683. static int msm_asoc_machine_remove(struct platform_device *pdev)
  684. {
  685. /* kobject_put decrease the kref count, once the count reaches 0.
  686. * Kobject core will automatically clean up the memory allocated by kobject.
  687. * The snd_card_sysfs_release release will help clean up memory allocated by us
  688. */
  689. kobject_put(&snd_card_pdata->snd_card_kobj);
  690. return 0;
  691. }
  692. static struct platform_driver gvm_asoc_machine_driver = {
  693. .driver = {
  694. .name = DRV_NAME,
  695. .pm = &snd_soc_pm_ops,
  696. .of_match_table = gvm_asoc_machine_of_match,
  697. },
  698. .probe = msm_asoc_machine_probe,
  699. .remove = msm_asoc_machine_remove,
  700. };
  701. int __init gvm_auto_spf_init(void)
  702. {
  703. return platform_driver_register(&gvm_asoc_machine_driver);
  704. }
  705. void gvm_auto_spf_exit(void)
  706. {
  707. platform_driver_unregister(&gvm_asoc_machine_driver);
  708. }
  709. module_init(gvm_auto_spf_init);
  710. module_exit(gvm_auto_spf_exit);
  711. MODULE_DESCRIPTION("ALSA SoC Machine Driver for SPF");
  712. MODULE_LICENSE("GPL");
  713. MODULE_ALIAS("platform:" DRV_NAME);
  714. MODULE_DEVICE_TABLE(of, gvm_asoc_machine_of_match);