msm-cdc-supply.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2016-2020, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/kernel.h>
  6. #include <linux/module.h>
  7. #include <linux/of_irq.h>
  8. #include <linux/of_device.h>
  9. #include <linux/slab.h>
  10. #include <linux/regulator/consumer.h>
  11. #include <asoc/msm-cdc-supply.h>
  12. #include <sound/soc.h>
  13. #define CODEC_DT_MAX_PROP_SIZE 40
  14. static int msm_cdc_dt_parse_vreg_info(struct device *dev,
  15. struct cdc_regulator *cdc_vreg,
  16. const char *name, bool is_ond)
  17. {
  18. char prop_name[CODEC_DT_MAX_PROP_SIZE];
  19. struct device_node *regulator_node = NULL;
  20. const __be32 *prop;
  21. int len, rc;
  22. u32 prop_val;
  23. /* Parse supply name */
  24. snprintf(prop_name, CODEC_DT_MAX_PROP_SIZE, "%s-supply", name);
  25. regulator_node = of_parse_phandle(dev->of_node, prop_name, 0);
  26. if (!regulator_node) {
  27. dev_err(dev, "%s: Looking up %s property in node %s failed",
  28. __func__, prop_name, dev->of_node->full_name);
  29. rc = -EINVAL;
  30. goto done;
  31. }
  32. cdc_vreg->name = name;
  33. cdc_vreg->ondemand = is_ond;
  34. /* Parse supply - voltage */
  35. snprintf(prop_name, CODEC_DT_MAX_PROP_SIZE, "qcom,%s-voltage", name);
  36. prop = of_get_property(dev->of_node, prop_name, &len);
  37. if (!prop || (len != (2 * sizeof(__be32)))) {
  38. dev_err(dev, "%s: %s %s property\n", __func__,
  39. prop ? "invalid format" : "no", prop_name);
  40. rc = -EINVAL;
  41. goto done;
  42. } else {
  43. cdc_vreg->min_uV = be32_to_cpup(&prop[0]);
  44. cdc_vreg->max_uV = be32_to_cpup(&prop[1]);
  45. }
  46. /* Parse supply - current */
  47. snprintf(prop_name, CODEC_DT_MAX_PROP_SIZE, "qcom,%s-current", name);
  48. rc = of_property_read_u32(dev->of_node, prop_name, &prop_val);
  49. if (rc) {
  50. dev_err(dev, "%s: Looking up %s property in node %s failed",
  51. __func__, prop_name, dev->of_node->full_name);
  52. goto done;
  53. }
  54. cdc_vreg->optimum_uA = prop_val;
  55. dev_info(dev, "%s: %s: vol=[%d %d]uV, curr=[%d]uA, ond %d\n",
  56. __func__, cdc_vreg->name, cdc_vreg->min_uV, cdc_vreg->max_uV,
  57. cdc_vreg->optimum_uA, cdc_vreg->ondemand);
  58. done:
  59. return rc;
  60. }
  61. static int msm_cdc_parse_supplies(struct device *dev,
  62. struct cdc_regulator *cdc_reg,
  63. const char *sup_list, int sup_cnt,
  64. bool is_ond)
  65. {
  66. int idx, rc = 0;
  67. const char *name = NULL;
  68. for (idx = 0; idx < sup_cnt; idx++) {
  69. rc = of_property_read_string_index(dev->of_node, sup_list, idx,
  70. &name);
  71. if (rc) {
  72. dev_err(dev, "%s: read string %s[%d] error (%d)\n",
  73. __func__, sup_list, idx, rc);
  74. goto done;
  75. }
  76. dev_dbg(dev, "%s: Found cdc supply %s as part of %s\n",
  77. __func__, name, sup_list);
  78. rc = msm_cdc_dt_parse_vreg_info(dev, &cdc_reg[idx], name,
  79. is_ond);
  80. if (rc) {
  81. dev_err(dev, "%s: parse %s vreg info failed (%d)\n",
  82. __func__, name, rc);
  83. goto done;
  84. }
  85. }
  86. done:
  87. return rc;
  88. }
  89. static int msm_cdc_check_supply_param(struct device *dev,
  90. struct cdc_regulator *cdc_vreg,
  91. int num_supplies)
  92. {
  93. if (!dev) {
  94. pr_err("%s: device is NULL\n", __func__);
  95. return -ENODEV;
  96. }
  97. if (!cdc_vreg || (num_supplies <= 0)) {
  98. dev_err(dev, "%s: supply check failed: vreg: %pK, num_supplies: %d\n",
  99. __func__, cdc_vreg, num_supplies);
  100. return -EINVAL;
  101. }
  102. return 0;
  103. }
  104. /*
  105. * msm_cdc_disable_ondemand_supply:
  106. * Disable codec ondemand supply
  107. *
  108. * @dev: pointer to codec device
  109. * @supplies: pointer to regulator bulk data
  110. * @cdc_vreg: pointer to platform regulator data
  111. * @num_supplies: number of supplies
  112. * @supply_name: Ondemand supply name to be enabled
  113. *
  114. * Return error code if supply disable is failed
  115. */
  116. int msm_cdc_disable_ondemand_supply(struct device *dev,
  117. struct regulator_bulk_data *supplies,
  118. struct cdc_regulator *cdc_vreg,
  119. int num_supplies,
  120. char *supply_name)
  121. {
  122. int rc, i;
  123. if ((!supply_name) || (!supplies)) {
  124. pr_err("%s: either dev or supplies or cdc_vreg is NULL\n",
  125. __func__);
  126. return -EINVAL;
  127. }
  128. /* input parameter validation */
  129. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  130. if (rc)
  131. return rc;
  132. for (i = 0; i < num_supplies; i++) {
  133. if (cdc_vreg[i].ondemand &&
  134. !strcmp(cdc_vreg[i].name, supply_name)) {
  135. rc = regulator_disable(supplies[i].consumer);
  136. if (rc)
  137. dev_err(dev, "%s: failed to disable supply %s, err:%d\n",
  138. __func__, supplies[i].supply, rc);
  139. break;
  140. }
  141. }
  142. if (i == num_supplies) {
  143. dev_err(dev, "%s: not able to find supply %s\n",
  144. __func__, supply_name);
  145. rc = -EINVAL;
  146. }
  147. return rc;
  148. }
  149. EXPORT_SYMBOL(msm_cdc_disable_ondemand_supply);
  150. /*
  151. * msm_cdc_enable_ondemand_supply:
  152. * Enable codec ondemand supply
  153. *
  154. * @dev: pointer to codec device
  155. * @supplies: pointer to regulator bulk data
  156. * @cdc_vreg: pointer to platform regulator data
  157. * @num_supplies: number of supplies
  158. * @supply_name: Ondemand supply name to be enabled
  159. *
  160. * Return error code if supply enable is failed
  161. */
  162. int msm_cdc_enable_ondemand_supply(struct device *dev,
  163. struct regulator_bulk_data *supplies,
  164. struct cdc_regulator *cdc_vreg,
  165. int num_supplies,
  166. char *supply_name)
  167. {
  168. int rc, i;
  169. if ((!supply_name) || (!supplies)) {
  170. pr_err("%s: either dev or supplies or cdc_vreg is NULL\n",
  171. __func__);
  172. return -EINVAL;
  173. }
  174. /* input parameter validation */
  175. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  176. if (rc)
  177. return rc;
  178. for (i = 0; i < num_supplies; i++) {
  179. if (cdc_vreg[i].ondemand &&
  180. !strcmp(cdc_vreg[i].name, supply_name)) {
  181. rc = regulator_enable(supplies[i].consumer);
  182. if (rc)
  183. dev_err(dev, "%s: failed to enable supply %s, rc: %d\n",
  184. __func__, supplies[i].supply, rc);
  185. break;
  186. }
  187. }
  188. if (i == num_supplies) {
  189. dev_err(dev, "%s: not able to find supply %s\n",
  190. __func__, supply_name);
  191. rc = -EINVAL;
  192. }
  193. return rc;
  194. }
  195. EXPORT_SYMBOL(msm_cdc_enable_ondemand_supply);
  196. /*
  197. * msm_cdc_disable_static_supplies:
  198. * Disable codec static supplies
  199. *
  200. * @dev: pointer to codec device
  201. * @supplies: pointer to regulator bulk data
  202. * @cdc_vreg: pointer to platform regulator data
  203. * @num_supplies: number of supplies
  204. *
  205. * Return error code if supply disable is failed
  206. */
  207. int msm_cdc_disable_static_supplies(struct device *dev,
  208. struct regulator_bulk_data *supplies,
  209. struct cdc_regulator *cdc_vreg,
  210. int num_supplies)
  211. {
  212. int rc, i;
  213. if ((!dev) || (!supplies) || (!cdc_vreg)) {
  214. pr_err("%s: either dev or supplies or cdc_vreg is NULL\n",
  215. __func__);
  216. return -EINVAL;
  217. }
  218. /* input parameter validation */
  219. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  220. if (rc)
  221. return rc;
  222. for (i = 0; i < num_supplies; i++) {
  223. if (cdc_vreg[i].ondemand)
  224. continue;
  225. rc = regulator_disable(supplies[i].consumer);
  226. if (rc)
  227. dev_err(dev, "%s: failed to disable supply %s, err:%d\n",
  228. __func__, supplies[i].supply, rc);
  229. else
  230. dev_dbg(dev, "%s: disabled regulator %s\n",
  231. __func__, supplies[i].supply);
  232. }
  233. return rc;
  234. }
  235. EXPORT_SYMBOL(msm_cdc_disable_static_supplies);
  236. /*
  237. * msm_cdc_release_supplies:
  238. * Release codec power supplies
  239. *
  240. * @dev: pointer to codec device
  241. * @supplies: pointer to regulator bulk data
  242. * @cdc_vreg: pointer to platform regulator data
  243. * @num_supplies: number of supplies
  244. *
  245. * Return error code if supply disable is failed
  246. */
  247. int msm_cdc_release_supplies(struct device *dev,
  248. struct regulator_bulk_data *supplies,
  249. struct cdc_regulator *cdc_vreg,
  250. int num_supplies)
  251. {
  252. int rc = 0;
  253. int i;
  254. if ((!dev) || (!supplies) || (!cdc_vreg)) {
  255. pr_err("%s: either dev or supplies or cdc_vreg is NULL\n",
  256. __func__);
  257. return -EINVAL;
  258. }
  259. /* input parameter validation */
  260. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  261. if (rc)
  262. return rc;
  263. msm_cdc_disable_static_supplies(dev, supplies, cdc_vreg,
  264. num_supplies);
  265. for (i = 0; i < num_supplies; i++) {
  266. if (regulator_count_voltages(supplies[i].consumer) < 0)
  267. continue;
  268. regulator_set_voltage(supplies[i].consumer, 0,
  269. cdc_vreg[i].max_uV);
  270. regulator_set_load(supplies[i].consumer, 0);
  271. }
  272. return rc;
  273. }
  274. EXPORT_SYMBOL(msm_cdc_release_supplies);
  275. /*
  276. * msm_cdc_enable_static_supplies:
  277. * Enable codec static supplies
  278. *
  279. * @dev: pointer to codec device
  280. * @supplies: pointer to regulator bulk data
  281. * @cdc_vreg: pointer to platform regulator data
  282. * @num_supplies: number of supplies
  283. *
  284. * Return error code if supply enable is failed
  285. */
  286. int msm_cdc_enable_static_supplies(struct device *dev,
  287. struct regulator_bulk_data *supplies,
  288. struct cdc_regulator *cdc_vreg,
  289. int num_supplies)
  290. {
  291. int rc, i;
  292. if ((!dev) || (!supplies) || (!cdc_vreg)) {
  293. pr_err("%s: either dev or supplies or cdc_vreg is NULL\n",
  294. __func__);
  295. return -EINVAL;
  296. }
  297. /* input parameter validation */
  298. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  299. if (rc)
  300. return rc;
  301. for (i = 0; i < num_supplies; i++) {
  302. if (cdc_vreg[i].ondemand)
  303. continue;
  304. rc = regulator_enable(supplies[i].consumer);
  305. if (rc) {
  306. dev_err(dev, "%s: failed to enable supply %s, rc: %d\n",
  307. __func__, supplies[i].supply, rc);
  308. break;
  309. }
  310. }
  311. while (rc && i--)
  312. if (!cdc_vreg[i].ondemand)
  313. regulator_disable(supplies[i].consumer);
  314. return rc;
  315. }
  316. EXPORT_SYMBOL(msm_cdc_enable_static_supplies);
  317. /*
  318. * msm_cdc_init_supplies:
  319. * Initialize codec static supplies
  320. *
  321. * @dev: pointer to codec device
  322. * @supplies: pointer to regulator bulk data
  323. * @cdc_vreg: pointer to platform regulator data
  324. * @num_supplies: number of supplies
  325. *
  326. * Return error code if supply init is failed
  327. */
  328. int msm_cdc_init_supplies(struct device *dev,
  329. struct regulator_bulk_data **supplies,
  330. struct cdc_regulator *cdc_vreg,
  331. int num_supplies)
  332. {
  333. return msm_cdc_init_supplies_v2(dev, supplies, cdc_vreg,
  334. num_supplies, false);
  335. }
  336. EXPORT_SYMBOL(msm_cdc_init_supplies);
  337. /*
  338. * msm_cdc_init_supplies_v2:
  339. * Initialize codec static supplies.
  340. * Initialize codec dynamic supplies based on vote_regulator_on_demand
  341. *
  342. * @dev: pointer to codec device
  343. * @supplies: pointer to regulator bulk data
  344. * @cdc_vreg: pointer to platform regulator data
  345. * @num_supplies: number of supplies
  346. * @vote_regulator_on_demand: initialize codec dynamic supplies at runtime
  347. *
  348. * Return error code if supply init is failed
  349. */
  350. int msm_cdc_init_supplies_v2(struct device *dev,
  351. struct regulator_bulk_data **supplies,
  352. struct cdc_regulator *cdc_vreg,
  353. int num_supplies, u32 vote_regulator_on_demand)
  354. {
  355. struct regulator_bulk_data *vsup;
  356. int rc;
  357. int i;
  358. if (!dev || !cdc_vreg) {
  359. pr_err("%s: device pointer or dce_vreg is NULL\n",
  360. __func__);
  361. return -EINVAL;
  362. }
  363. /* input parameter validation */
  364. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  365. if (rc)
  366. return rc;
  367. vsup = devm_kcalloc(dev, num_supplies,
  368. sizeof(struct regulator_bulk_data),
  369. GFP_KERNEL);
  370. if (!vsup)
  371. return -ENOMEM;
  372. for (i = 0; i < num_supplies; i++) {
  373. if (!cdc_vreg[i].name) {
  374. dev_err(dev, "%s: supply name not defined\n",
  375. __func__);
  376. rc = -EINVAL;
  377. goto err_supply;
  378. }
  379. vsup[i].supply = cdc_vreg[i].name;
  380. }
  381. rc = devm_regulator_bulk_get(dev, num_supplies, vsup);
  382. if (rc) {
  383. dev_err(dev, "%s: failed to get supplies (%d)\n",
  384. __func__, rc);
  385. goto err_supply;
  386. }
  387. /* Set voltage and current on regulators */
  388. for (i = 0; i < num_supplies; i++) {
  389. if (regulator_count_voltages(vsup[i].consumer) < 0)
  390. continue;
  391. if (cdc_vreg[i].ondemand && vote_regulator_on_demand)
  392. continue;
  393. rc = regulator_set_voltage(vsup[i].consumer,
  394. cdc_vreg[i].min_uV,
  395. cdc_vreg[i].max_uV);
  396. if (rc) {
  397. dev_err(dev, "%s: set regulator voltage failed for %s, err:%d\n",
  398. __func__, vsup[i].supply, rc);
  399. goto err_supply;
  400. }
  401. rc = regulator_set_load(vsup[i].consumer,
  402. cdc_vreg[i].optimum_uA);
  403. if (rc < 0) {
  404. dev_err(dev, "%s: set regulator optimum mode failed for %s, err:%d\n",
  405. __func__, vsup[i].supply, rc);
  406. goto err_supply;
  407. }
  408. }
  409. *supplies = vsup;
  410. return 0;
  411. err_supply:
  412. return rc;
  413. }
  414. EXPORT_SYMBOL(msm_cdc_init_supplies_v2);
  415. /*
  416. * msm_cdc_get_power_supplies:
  417. * Get codec power supplies from device tree.
  418. * Allocate memory to hold regulator data for
  419. * all power supplies.
  420. *
  421. * @dev: pointer to codec device
  422. * @cdc_vreg: pointer to codec regulator
  423. * @total_num_supplies: total number of supplies read from DT
  424. *
  425. * Return error code if supply disable is failed
  426. */
  427. int msm_cdc_get_power_supplies(struct device *dev,
  428. struct cdc_regulator **cdc_vreg,
  429. int *total_num_supplies)
  430. {
  431. const char *static_prop_name = "qcom,cdc-static-supplies";
  432. const char *ond_prop_name = "qcom,cdc-on-demand-supplies";
  433. const char *cp_prop_name = "qcom,cdc-cp-supplies";
  434. int static_sup_cnt = 0;
  435. int ond_sup_cnt = 0;
  436. int cp_sup_cnt = 0;
  437. int num_supplies = 0;
  438. struct cdc_regulator *cdc_reg;
  439. int rc;
  440. if (!dev) {
  441. pr_err("%s: device pointer is NULL\n", __func__);
  442. return -EINVAL;
  443. }
  444. static_sup_cnt = of_property_count_strings(dev->of_node,
  445. static_prop_name);
  446. if (static_sup_cnt < 0) {
  447. dev_err(dev, "%s: Failed to get static supplies(%d)\n",
  448. __func__, static_sup_cnt);
  449. rc = static_sup_cnt;
  450. goto err_supply_cnt;
  451. }
  452. ond_sup_cnt = of_property_count_strings(dev->of_node, ond_prop_name);
  453. if (ond_sup_cnt < 0)
  454. ond_sup_cnt = 0;
  455. cp_sup_cnt = of_property_count_strings(dev->of_node,
  456. cp_prop_name);
  457. if (cp_sup_cnt < 0)
  458. cp_sup_cnt = 0;
  459. num_supplies = static_sup_cnt + ond_sup_cnt + cp_sup_cnt;
  460. if (num_supplies <= 0) {
  461. dev_err(dev, "%s: supply count is 0 or negative\n", __func__);
  462. rc = -EINVAL;
  463. goto err_supply_cnt;
  464. }
  465. cdc_reg = devm_kcalloc(dev, num_supplies,
  466. sizeof(struct cdc_regulator),
  467. GFP_KERNEL);
  468. if (!cdc_reg) {
  469. rc = -ENOMEM;
  470. goto err_mem_alloc;
  471. }
  472. rc = msm_cdc_parse_supplies(dev, cdc_reg, static_prop_name,
  473. static_sup_cnt, false);
  474. if (rc) {
  475. dev_err(dev, "%s: failed to parse static supplies(%d)\n",
  476. __func__, rc);
  477. goto err_sup;
  478. }
  479. rc = msm_cdc_parse_supplies(dev, &cdc_reg[static_sup_cnt],
  480. ond_prop_name, ond_sup_cnt,
  481. true);
  482. if (rc) {
  483. dev_err(dev, "%s: failed to parse demand supplies(%d)\n",
  484. __func__, rc);
  485. goto err_sup;
  486. }
  487. rc = msm_cdc_parse_supplies(dev,
  488. &cdc_reg[static_sup_cnt + ond_sup_cnt],
  489. cp_prop_name, cp_sup_cnt, true);
  490. if (rc) {
  491. dev_err(dev, "%s: failed to parse cp supplies(%d)\n",
  492. __func__, rc);
  493. goto err_sup;
  494. }
  495. *cdc_vreg = cdc_reg;
  496. *total_num_supplies = num_supplies;
  497. return 0;
  498. err_sup:
  499. err_supply_cnt:
  500. err_mem_alloc:
  501. return rc;
  502. }
  503. EXPORT_SYMBOL(msm_cdc_get_power_supplies);
  504. /*
  505. * msm_cdc_init_wcd_supply:
  506. * Initialize wcd supply parameters.
  507. *
  508. * @np: device node pointer to codec device
  509. * @name: power supply name
  510. * @cdc_supply: codec supply struct to hold wcd params
  511. *
  512. * Return error code if init failed
  513. */
  514. int msm_cdc_init_wcd_supply(struct device_node *np, const char *name,
  515. struct cdc_wcd_supply *cdc_supply)
  516. {
  517. struct platform_device *pdev = NULL;
  518. if (!np || !cdc_supply)
  519. return -EINVAL;
  520. pdev = of_find_device_by_node(np);
  521. if (!pdev)
  522. return -EINVAL;
  523. cdc_supply->dev = &pdev->dev;
  524. cdc_supply->name = name;
  525. cdc_supply->component = snd_soc_lookup_component(&pdev->dev, NULL);
  526. return 0;
  527. }
  528. EXPORT_SYMBOL(msm_cdc_init_wcd_supply);
  529. /*
  530. * msm_cdc_enable_wcd_supply:
  531. * Enable/Disable wcd supply.
  532. *
  533. * @cdc_supply: codec supply struct to hold wcd params
  534. * @enable: bool to inform whether to enable or disable
  535. *
  536. * Return error code if enable/disable failed
  537. */
  538. int msm_cdc_enable_wcd_supply(struct cdc_wcd_supply *cdc_supply, bool enable)
  539. {
  540. struct snd_soc_component *component = cdc_supply->component;
  541. int rc;
  542. if (!component) {
  543. pr_err("%s: Component memory is NULL\n", __func__);
  544. return -EINVAL;
  545. }
  546. if (enable)
  547. rc = snd_soc_dapm_force_enable_pin(
  548. snd_soc_component_get_dapm(component),
  549. cdc_supply->name);
  550. else
  551. rc = snd_soc_dapm_disable_pin(
  552. snd_soc_component_get_dapm(component),
  553. cdc_supply->name);
  554. if (!rc)
  555. snd_soc_dapm_sync(snd_soc_component_get_dapm(component));
  556. else
  557. dev_err(component->dev, "%s: micbias %s force %s pin failed\n",
  558. __func__, cdc_supply->name, (enable ? "enable" : "disable"));
  559. return rc;
  560. }
  561. EXPORT_SYMBOL(msm_cdc_enable_wcd_supply);