gvm_auto_spf_dummy.c 21 KB

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