msm-cdc-supply.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2016-2020, The Linux Foundation. All rights reserved.
  4. * Copyright (c) 2023, Qualcomm Innovation Center, Inc. All rights reserved.
  5. */
  6. #include <linux/kernel.h>
  7. #include <linux/module.h>
  8. #include <linux/of_irq.h>
  9. #include <linux/of_device.h>
  10. #include <linux/slab.h>
  11. #include <linux/regulator/consumer.h>
  12. #include <asoc/msm-cdc-supply.h>
  13. #include <sound/soc.h>
  14. #define CODEC_DT_MAX_PROP_SIZE 40
  15. static int msm_cdc_dt_parse_vreg_info(struct device *dev,
  16. struct cdc_regulator *cdc_vreg,
  17. const char *name, bool is_ond)
  18. {
  19. char prop_name[CODEC_DT_MAX_PROP_SIZE];
  20. struct device_node *regulator_node = NULL;
  21. const __be32 *prop;
  22. int len, rc;
  23. u32 prop_val;
  24. /* Parse supply name */
  25. snprintf(prop_name, CODEC_DT_MAX_PROP_SIZE, "%s-supply", name);
  26. regulator_node = of_parse_phandle(dev->of_node, prop_name, 0);
  27. if (!regulator_node) {
  28. dev_err(dev, "%s: Looking up %s property in node %s failed",
  29. __func__, prop_name, dev->of_node->full_name);
  30. rc = -EINVAL;
  31. goto done;
  32. }
  33. cdc_vreg->name = name;
  34. cdc_vreg->ondemand = is_ond;
  35. /* Parse supply - voltage */
  36. snprintf(prop_name, CODEC_DT_MAX_PROP_SIZE, "qcom,%s-voltage", name);
  37. prop = of_get_property(dev->of_node, prop_name, &len);
  38. if (!prop || (len != (2 * sizeof(__be32)))) {
  39. dev_err(dev, "%s: %s %s property\n", __func__,
  40. prop ? "invalid format" : "no", prop_name);
  41. rc = -EINVAL;
  42. goto done;
  43. } else {
  44. cdc_vreg->min_uV = be32_to_cpup(&prop[0]);
  45. cdc_vreg->max_uV = be32_to_cpup(&prop[1]);
  46. }
  47. /* Parse supply - current */
  48. snprintf(prop_name, CODEC_DT_MAX_PROP_SIZE, "qcom,%s-current", name);
  49. rc = of_property_read_u32(dev->of_node, prop_name, &prop_val);
  50. if (rc) {
  51. dev_err(dev, "%s: Looking up %s property in node %s failed",
  52. __func__, prop_name, dev->of_node->full_name);
  53. goto done;
  54. }
  55. cdc_vreg->optimum_uA = prop_val;
  56. /* Parse supply - LPM or NOM mode(default NOM) */
  57. snprintf(prop_name, CODEC_DT_MAX_PROP_SIZE, "qcom,%s-lpm-supported", name);
  58. rc = of_property_read_u32(dev->of_node, prop_name, &prop_val);
  59. if (rc) {
  60. dev_dbg(dev, "%s: Looking up %s property in node %s failed",
  61. __func__, prop_name, dev->of_node->full_name);
  62. cdc_vreg->lpm_supported = 0;
  63. rc = 0;
  64. } else {
  65. cdc_vreg->lpm_supported = prop_val;
  66. }
  67. /* Parse supply - retention mode */
  68. snprintf(prop_name, CODEC_DT_MAX_PROP_SIZE, "qcom,%s-rem-supported", name);
  69. rc = of_property_read_u32(dev->of_node, prop_name, &prop_val);
  70. if (rc) {
  71. dev_dbg(dev, "%s: Looking up %s property in node %s failed",
  72. __func__, prop_name, dev->of_node->full_name);
  73. cdc_vreg->rem_supported = 0;
  74. rc = 0;
  75. } else {
  76. cdc_vreg->rem_supported = prop_val;
  77. }
  78. dev_info(dev, "%s: %s: vol=[%d %d]uV, curr=[%d]uA, ond %d lpm %d rem %d\n",
  79. __func__, cdc_vreg->name, cdc_vreg->min_uV, cdc_vreg->max_uV,
  80. cdc_vreg->optimum_uA, cdc_vreg->ondemand,
  81. cdc_vreg->lpm_supported, cdc_vreg->rem_supported);
  82. done:
  83. return rc;
  84. }
  85. static int msm_cdc_parse_supplies(struct device *dev,
  86. struct cdc_regulator *cdc_reg,
  87. const char *sup_list, int sup_cnt,
  88. bool is_ond)
  89. {
  90. int idx, rc = 0;
  91. const char *name = NULL;
  92. for (idx = 0; idx < sup_cnt; idx++) {
  93. rc = of_property_read_string_index(dev->of_node, sup_list, idx,
  94. &name);
  95. if (rc) {
  96. dev_err(dev, "%s: read string %s[%d] error (%d)\n",
  97. __func__, sup_list, idx, rc);
  98. goto done;
  99. }
  100. dev_dbg(dev, "%s: Found cdc supply %s as part of %s\n",
  101. __func__, name, sup_list);
  102. rc = msm_cdc_dt_parse_vreg_info(dev, &cdc_reg[idx], name,
  103. is_ond);
  104. if (rc) {
  105. dev_err(dev, "%s: parse %s vreg info failed (%d)\n",
  106. __func__, name, rc);
  107. goto done;
  108. }
  109. }
  110. done:
  111. return rc;
  112. }
  113. static int msm_cdc_check_supply_param(struct device *dev,
  114. struct cdc_regulator *cdc_vreg,
  115. int num_supplies)
  116. {
  117. if (!dev) {
  118. pr_err_ratelimited("%s: device is NULL\n", __func__);
  119. return -ENODEV;
  120. }
  121. if (!cdc_vreg || (num_supplies <= 0)) {
  122. dev_err_ratelimited(dev, "%s: supply check failed: vreg: %pK, num_supplies: %d\n",
  123. __func__, cdc_vreg, num_supplies);
  124. return -EINVAL;
  125. }
  126. return 0;
  127. }
  128. /*
  129. * msm_cdc_is_ondemand_supply:
  130. * return if ondemand supply true or not
  131. *
  132. * @dev: pointer to codec device
  133. * @supplies: pointer to regulator bulk data
  134. * @cdc_vreg: pointer to platform regulator data
  135. * @num_supplies: number of supplies
  136. * @supply_name: supply name to be checked
  137. *
  138. * Return true/false
  139. */
  140. bool msm_cdc_is_ondemand_supply(struct device *dev,
  141. struct regulator_bulk_data *supplies,
  142. struct cdc_regulator *cdc_vreg,
  143. int num_supplies,
  144. char *supply_name)
  145. {
  146. bool rc = false;
  147. int ret, i;
  148. if ((!supply_name) || (!supplies)) {
  149. pr_err_ratelimited("%s: either dev or supplies or cdc_vreg is NULL\n",
  150. __func__);
  151. return rc;
  152. }
  153. /* input parameter validation */
  154. ret = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  155. if (ret)
  156. return rc;
  157. for (i = 0; i < num_supplies; i++) {
  158. if (cdc_vreg[i].ondemand &&
  159. !strcmp(cdc_vreg[i].name, supply_name))
  160. return true;
  161. }
  162. return rc;
  163. }
  164. EXPORT_SYMBOL(msm_cdc_is_ondemand_supply);
  165. /*
  166. * msm_cdc_supply_supports_retention_mode:
  167. * On certain hardware configurations, This means that the
  168. * PM will disable the supply and remove its power vote
  169. * if the PM enters into a suspended state.
  170. *
  171. * return if supply supports retention mode or not
  172. *
  173. * @dev: pointer to codec device
  174. * @supplies: pointer to regulator bulk data
  175. * @cdc_vreg: pointer to platform regulator data
  176. * @num_supplies: number of supplies
  177. * @supply_name: supply name to be checked
  178. *
  179. * Return true/false
  180. */
  181. bool msm_cdc_supply_supports_retention_mode(struct device *dev,
  182. struct regulator_bulk_data *supplies,
  183. struct cdc_regulator *cdc_vreg,
  184. int num_supplies, char *supply_name)
  185. {
  186. bool rc = false;
  187. int ret, i;
  188. if ((!supply_name) || (!supplies)) {
  189. pr_err_ratelimited("%s: either dev or supplies or cdc_vreg is NULL\n",
  190. __func__);
  191. return rc;
  192. }
  193. /* input parameter validation */
  194. ret = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  195. if (ret)
  196. return rc;
  197. for (i = 0; i < num_supplies; i++) {
  198. if (cdc_vreg[i].rem_supported &&
  199. !strcmp(cdc_vreg[i].name, supply_name))
  200. return true;
  201. }
  202. return rc;
  203. }
  204. EXPORT_SYMBOL(msm_cdc_supply_supports_retention_mode);
  205. /*
  206. * msm_cdc_check_supply_vote:
  207. *
  208. * return true if supply has voted for regulator enable
  209. *
  210. * @dev: pointer to codec device
  211. * @supplies: pointer to regulator bulk data
  212. * @cdc_vreg: pointer to platform regulator data
  213. * @num_supplies: number of supplies
  214. * @supply_name: supply name to be checked
  215. *
  216. * Return true/false
  217. */
  218. bool msm_cdc_check_supply_vote(struct device *dev,
  219. struct regulator_bulk_data *supplies,
  220. struct cdc_regulator *cdc_vreg,
  221. int num_supplies,
  222. char *supply_name)
  223. {
  224. bool rc = false;
  225. int ret, i;
  226. if ((!supply_name) || (!supplies)) {
  227. pr_err_ratelimited("%s: either dev or supplies or cdc_vreg is NULL\n",
  228. __func__);
  229. return rc;
  230. }
  231. /* input parameter validation */
  232. ret = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  233. if (ret)
  234. return rc;
  235. for (i = 0; i < num_supplies; i++) {
  236. if (strcmp(cdc_vreg[i].name, supply_name) != 0)
  237. continue;
  238. return cdc_vreg[i].vote;
  239. }
  240. return rc;
  241. }
  242. EXPORT_SYMBOL(msm_cdc_check_supply_vote);
  243. /*
  244. * msm_cdc_set_supply_min_voltage:
  245. * Set min supply voltage for particular supply
  246. *
  247. * @dev: pointer to codec device
  248. * @supplies: pointer to regulator bulk data
  249. * @cdc_vreg: pointer to platform regulator data
  250. * @num_supplies: number of supplies
  251. * @supply_name: Supply name to change voltage for
  252. * @vval_min: Min voltage to be set in uV
  253. * @override_min_vol: True if override min voltage from default
  254. * Return error code if unable to set voltage
  255. */
  256. int msm_cdc_set_supply_min_voltage(struct device *dev,
  257. struct regulator_bulk_data *supplies,
  258. struct cdc_regulator *cdc_vreg,
  259. int num_supplies, char *supply_name,
  260. int vval_min, bool override_min_vol)
  261. {
  262. int rc = 0, i;
  263. if ((!supply_name) || (!supplies)) {
  264. pr_err_ratelimited("%s: either dev or supplies or cdc_vreg is NULL\n",
  265. __func__);
  266. return -EINVAL;
  267. }
  268. /* input parameter validation */
  269. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  270. if (rc)
  271. return rc;
  272. for (i = 0; i < num_supplies; i++) {
  273. if (!strcmp(cdc_vreg[i].name, supply_name)) {
  274. if (override_min_vol)
  275. regulator_set_voltage(supplies[i].consumer,
  276. vval_min, cdc_vreg[i].max_uV);
  277. else
  278. regulator_set_voltage(supplies[i].consumer,
  279. cdc_vreg[i].min_uV, cdc_vreg[i].max_uV);
  280. break;
  281. }
  282. }
  283. return rc;
  284. }
  285. EXPORT_SYMBOL(msm_cdc_set_supply_min_voltage);
  286. /*
  287. * msm_cdc_disable_ondemand_supply:
  288. * Disable codec ondemand supply
  289. *
  290. * @dev: pointer to codec device
  291. * @supplies: pointer to regulator bulk data
  292. * @cdc_vreg: pointer to platform regulator data
  293. * @num_supplies: number of supplies
  294. * @supply_name: Ondemand supply name to be enabled
  295. *
  296. * Return error code if supply disable is failed
  297. */
  298. int msm_cdc_disable_ondemand_supply(struct device *dev,
  299. struct regulator_bulk_data *supplies,
  300. struct cdc_regulator *cdc_vreg,
  301. int num_supplies,
  302. char *supply_name)
  303. {
  304. int rc, i;
  305. if ((!supply_name) || (!supplies)) {
  306. pr_err_ratelimited("%s: either dev or supplies or cdc_vreg is NULL\n",
  307. __func__);
  308. return -EINVAL;
  309. }
  310. /* input parameter validation */
  311. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  312. if (rc)
  313. return rc;
  314. for (i = 0; i < num_supplies; i++) {
  315. if (cdc_vreg[i].ondemand &&
  316. !strcmp(cdc_vreg[i].name, supply_name)) {
  317. rc = regulator_disable(supplies[i].consumer);
  318. if (rc)
  319. dev_err_ratelimited(dev,
  320. "%s: failed to disable supply %s, err:%d\n",
  321. __func__, supplies[i].supply, rc);
  322. cdc_vreg[i].vote = false;
  323. break;
  324. }
  325. }
  326. if (i == num_supplies) {
  327. dev_err_ratelimited(dev, "%s: not able to find supply %s\n",
  328. __func__, supply_name);
  329. rc = -EINVAL;
  330. }
  331. return rc;
  332. }
  333. EXPORT_SYMBOL(msm_cdc_disable_ondemand_supply);
  334. /*
  335. * msm_cdc_enable_ondemand_supply:
  336. * Enable codec ondemand supply
  337. *
  338. * @dev: pointer to codec device
  339. * @supplies: pointer to regulator bulk data
  340. * @cdc_vreg: pointer to platform regulator data
  341. * @num_supplies: number of supplies
  342. * @supply_name: Ondemand supply name to be enabled
  343. *
  344. * Return error code if supply enable is failed
  345. */
  346. int msm_cdc_enable_ondemand_supply(struct device *dev,
  347. struct regulator_bulk_data *supplies,
  348. struct cdc_regulator *cdc_vreg,
  349. int num_supplies,
  350. char *supply_name)
  351. {
  352. int rc, i;
  353. if ((!supply_name) || (!supplies)) {
  354. pr_err_ratelimited("%s: either dev or supplies or cdc_vreg is NULL\n",
  355. __func__);
  356. return -EINVAL;
  357. }
  358. /* input parameter validation */
  359. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  360. if (rc)
  361. return rc;
  362. for (i = 0; i < num_supplies; i++) {
  363. if (cdc_vreg[i].ondemand &&
  364. !strcmp(cdc_vreg[i].name, supply_name)) {
  365. rc = regulator_enable(supplies[i].consumer);
  366. if (rc)
  367. dev_err_ratelimited(dev, "%s: failed to enable supply %s, rc: %d\n",
  368. __func__, supplies[i].supply, rc);
  369. cdc_vreg[i].vote = true;
  370. break;
  371. }
  372. }
  373. if (i == num_supplies) {
  374. dev_err_ratelimited(dev, "%s: not able to find supply %s\n",
  375. __func__, supply_name);
  376. rc = -EINVAL;
  377. }
  378. return rc;
  379. }
  380. EXPORT_SYMBOL(msm_cdc_enable_ondemand_supply);
  381. /*
  382. * msm_cdc_set_supplies_lpm_mode:
  383. * Update load for given supply string
  384. *
  385. * @dev: pointer to codec device
  386. * @supplies: pointer to regulator bulk data
  387. * @cdc_vreg: pointer to platform regulator data
  388. * @num_supplies: number of supplies
  389. * @supply_name: supply name to be checked
  390. * @min_max: Apply optimum or 0 current
  391. *
  392. * Return error code if set current fail
  393. */
  394. int msm_cdc_set_supplies_lpm_mode(struct device *dev,
  395. struct regulator_bulk_data *supplies,
  396. struct cdc_regulator *cdc_vreg,
  397. int num_supplies,
  398. bool flag)
  399. {
  400. int rc = 0, i;
  401. if (!supplies) {
  402. pr_err_ratelimited("%s: supplies is NULL\n",
  403. __func__);
  404. return -EINVAL;
  405. }
  406. /* input parameter validation */
  407. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  408. if (rc)
  409. return rc;
  410. for (i = 0; i < num_supplies; i++) {
  411. if (cdc_vreg[i].lpm_supported) {
  412. rc = regulator_set_load(
  413. supplies[i].consumer,
  414. flag ? 0 : cdc_vreg[i].optimum_uA);
  415. if (rc)
  416. dev_err_ratelimited(dev,
  417. "%s: failed to set supply %s to %s, err:%d\n",
  418. __func__, supplies[i].supply,
  419. flag ? "LPM" : "NOM",
  420. rc);
  421. else
  422. dev_dbg(dev, "%s: regulator %s load set to %s\n",
  423. __func__, supplies[i].supply,
  424. flag ? "LPM" : "NOM");
  425. }
  426. }
  427. return rc;
  428. }
  429. EXPORT_SYMBOL(msm_cdc_set_supplies_lpm_mode);
  430. /*
  431. * msm_cdc_disable_static_supplies:
  432. * Disable codec static supplies
  433. *
  434. * @dev: pointer to codec device
  435. * @supplies: pointer to regulator bulk data
  436. * @cdc_vreg: pointer to platform regulator data
  437. * @num_supplies: number of supplies
  438. *
  439. * Return error code if supply disable is failed
  440. */
  441. int msm_cdc_disable_static_supplies(struct device *dev,
  442. struct regulator_bulk_data *supplies,
  443. struct cdc_regulator *cdc_vreg,
  444. int num_supplies)
  445. {
  446. int rc, i;
  447. if ((!dev) || (!supplies) || (!cdc_vreg)) {
  448. pr_err("%s: either dev or supplies or cdc_vreg is NULL\n",
  449. __func__);
  450. return -EINVAL;
  451. }
  452. /* input parameter validation */
  453. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  454. if (rc)
  455. return rc;
  456. for (i = 0; i < num_supplies; i++) {
  457. if (cdc_vreg[i].ondemand)
  458. continue;
  459. rc = regulator_disable(supplies[i].consumer);
  460. if (rc)
  461. dev_err(dev, "%s: failed to disable supply %s, err:%d\n",
  462. __func__, supplies[i].supply, rc);
  463. else {
  464. cdc_vreg[i].vote = false;
  465. dev_dbg(dev, "%s: disabled regulator %s\n",
  466. __func__, supplies[i].supply);
  467. }
  468. }
  469. return rc;
  470. }
  471. EXPORT_SYMBOL(msm_cdc_disable_static_supplies);
  472. /*
  473. * msm_cdc_release_supplies:
  474. * Release codec power supplies
  475. *
  476. * @dev: pointer to codec device
  477. * @supplies: pointer to regulator bulk data
  478. * @cdc_vreg: pointer to platform regulator data
  479. * @num_supplies: number of supplies
  480. *
  481. * Return error code if supply disable is failed
  482. */
  483. int msm_cdc_release_supplies(struct device *dev,
  484. struct regulator_bulk_data *supplies,
  485. struct cdc_regulator *cdc_vreg,
  486. int num_supplies)
  487. {
  488. int rc = 0;
  489. int i;
  490. if ((!dev) || (!supplies) || (!cdc_vreg)) {
  491. pr_err("%s: either dev or supplies or cdc_vreg is NULL\n",
  492. __func__);
  493. return -EINVAL;
  494. }
  495. /* input parameter validation */
  496. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  497. if (rc)
  498. return rc;
  499. msm_cdc_disable_static_supplies(dev, supplies, cdc_vreg,
  500. num_supplies);
  501. for (i = 0; i < num_supplies; i++) {
  502. if (regulator_count_voltages(supplies[i].consumer) < 0)
  503. continue;
  504. regulator_set_voltage(supplies[i].consumer, 0,
  505. cdc_vreg[i].max_uV);
  506. regulator_set_load(supplies[i].consumer, 0);
  507. }
  508. return rc;
  509. }
  510. EXPORT_SYMBOL(msm_cdc_release_supplies);
  511. /*
  512. * msm_cdc_enable_static_supplies:
  513. * Enable codec static supplies
  514. *
  515. * @dev: pointer to codec device
  516. * @supplies: pointer to regulator bulk data
  517. * @cdc_vreg: pointer to platform regulator data
  518. * @num_supplies: number of supplies
  519. *
  520. * Return error code if supply enable is failed
  521. */
  522. int msm_cdc_enable_static_supplies(struct device *dev,
  523. struct regulator_bulk_data *supplies,
  524. struct cdc_regulator *cdc_vreg,
  525. int num_supplies)
  526. {
  527. int rc, i;
  528. if ((!dev) || (!supplies) || (!cdc_vreg)) {
  529. pr_err("%s: either dev or supplies or cdc_vreg is NULL\n",
  530. __func__);
  531. return -EINVAL;
  532. }
  533. /* input parameter validation */
  534. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  535. if (rc)
  536. return rc;
  537. for (i = 0; i < num_supplies; i++) {
  538. if (cdc_vreg[i].ondemand)
  539. continue;
  540. rc = regulator_enable(supplies[i].consumer);
  541. if (rc) {
  542. dev_err(dev, "%s: failed to enable supply %s, rc: %d\n",
  543. __func__, supplies[i].supply, rc);
  544. break;
  545. }
  546. cdc_vreg[i].vote = true;
  547. }
  548. if (rc) {
  549. while (i--) {
  550. if (cdc_vreg[i].ondemand)
  551. continue;
  552. if (regulator_disable(supplies[i].consumer) == 0)
  553. cdc_vreg[i].vote = false;
  554. else
  555. dev_err(dev, "%s: failed to disable supply %s during unwind\n",
  556. __func__, supplies[i].supply);
  557. }
  558. }
  559. return rc;
  560. }
  561. EXPORT_SYMBOL(msm_cdc_enable_static_supplies);
  562. /*
  563. * msm_cdc_init_supplies:
  564. * Initialize codec static supplies
  565. *
  566. * @dev: pointer to codec device
  567. * @supplies: pointer to regulator bulk data
  568. * @cdc_vreg: pointer to platform regulator data
  569. * @num_supplies: number of supplies
  570. *
  571. * Return error code if supply init is failed
  572. */
  573. int msm_cdc_init_supplies(struct device *dev,
  574. struct regulator_bulk_data **supplies,
  575. struct cdc_regulator *cdc_vreg,
  576. int num_supplies)
  577. {
  578. return msm_cdc_init_supplies_v2(dev, supplies, cdc_vreg,
  579. num_supplies, false);
  580. }
  581. EXPORT_SYMBOL(msm_cdc_init_supplies);
  582. /*
  583. * msm_cdc_init_supplies_v2:
  584. * Initialize codec static supplies.
  585. * Initialize codec dynamic supplies based on vote_regulator_on_demand
  586. *
  587. * @dev: pointer to codec device
  588. * @supplies: pointer to regulator bulk data
  589. * @cdc_vreg: pointer to platform regulator data
  590. * @num_supplies: number of supplies
  591. * @vote_regulator_on_demand: initialize codec dynamic supplies at runtime
  592. *
  593. * Return error code if supply init is failed
  594. */
  595. int msm_cdc_init_supplies_v2(struct device *dev,
  596. struct regulator_bulk_data **supplies,
  597. struct cdc_regulator *cdc_vreg,
  598. int num_supplies, u32 vote_regulator_on_demand)
  599. {
  600. struct regulator_bulk_data *vsup;
  601. int rc;
  602. int i;
  603. if (!dev || !cdc_vreg) {
  604. pr_err("%s: device pointer or dce_vreg is NULL\n",
  605. __func__);
  606. return -EINVAL;
  607. }
  608. /* input parameter validation */
  609. rc = msm_cdc_check_supply_param(dev, cdc_vreg, num_supplies);
  610. if (rc)
  611. return rc;
  612. vsup = devm_kcalloc(dev, num_supplies,
  613. sizeof(struct regulator_bulk_data),
  614. GFP_KERNEL);
  615. if (!vsup)
  616. return -ENOMEM;
  617. for (i = 0; i < num_supplies; i++) {
  618. if (!cdc_vreg[i].name) {
  619. dev_err(dev, "%s: supply name not defined\n",
  620. __func__);
  621. rc = -EINVAL;
  622. goto err_supply;
  623. }
  624. vsup[i].supply = cdc_vreg[i].name;
  625. }
  626. rc = devm_regulator_bulk_get(dev, num_supplies, vsup);
  627. if (rc) {
  628. dev_err(dev, "%s: failed to get supplies (%d)\n",
  629. __func__, rc);
  630. goto err_supply;
  631. }
  632. /* Set voltage and current on regulators */
  633. for (i = 0; i < num_supplies; i++) {
  634. if (regulator_count_voltages(vsup[i].consumer) < 0)
  635. continue;
  636. if (cdc_vreg[i].ondemand && vote_regulator_on_demand)
  637. continue;
  638. cdc_vreg[i].regulator = vsup[i].consumer;
  639. rc = regulator_set_voltage(vsup[i].consumer,
  640. cdc_vreg[i].min_uV,
  641. cdc_vreg[i].max_uV);
  642. if (rc) {
  643. dev_err(dev, "%s: set regulator voltage failed for %s, err:%d\n",
  644. __func__, vsup[i].supply, rc);
  645. goto err_supply;
  646. }
  647. rc = regulator_set_load(vsup[i].consumer,
  648. cdc_vreg[i].optimum_uA);
  649. if (rc < 0) {
  650. dev_err(dev, "%s: set regulator optimum mode failed for %s, err:%d\n",
  651. __func__, vsup[i].supply, rc);
  652. goto err_supply;
  653. }
  654. }
  655. *supplies = vsup;
  656. return 0;
  657. err_supply:
  658. return rc;
  659. }
  660. EXPORT_SYMBOL(msm_cdc_init_supplies_v2);
  661. /*
  662. * msm_cdc_get_power_supplies:
  663. * Get codec power supplies from device tree.
  664. * Allocate memory to hold regulator data for
  665. * all power supplies.
  666. *
  667. * @dev: pointer to codec device
  668. * @cdc_vreg: pointer to codec regulator
  669. * @total_num_supplies: total number of supplies read from DT
  670. *
  671. * Return error code if supply disable is failed
  672. */
  673. int msm_cdc_get_power_supplies(struct device *dev,
  674. struct cdc_regulator **cdc_vreg,
  675. int *total_num_supplies)
  676. {
  677. const char *static_prop_name = "qcom,cdc-static-supplies";
  678. const char *ond_prop_name = "qcom,cdc-on-demand-supplies";
  679. const char *cp_prop_name = "qcom,cdc-cp-supplies";
  680. int static_sup_cnt = 0;
  681. int ond_sup_cnt = 0;
  682. int cp_sup_cnt = 0;
  683. int num_supplies = 0;
  684. struct cdc_regulator *cdc_reg;
  685. int rc;
  686. if (!dev) {
  687. pr_err_ratelimited("%s: device pointer is NULL\n", __func__);
  688. return -EINVAL;
  689. }
  690. static_sup_cnt = of_property_count_strings(dev->of_node,
  691. static_prop_name);
  692. if (static_sup_cnt < 0) {
  693. dev_err_ratelimited(dev, "%s: Failed to get static supplies(%d)\n",
  694. __func__, static_sup_cnt);
  695. rc = static_sup_cnt;
  696. goto err_supply_cnt;
  697. }
  698. ond_sup_cnt = of_property_count_strings(dev->of_node, ond_prop_name);
  699. if (ond_sup_cnt < 0)
  700. ond_sup_cnt = 0;
  701. cp_sup_cnt = of_property_count_strings(dev->of_node,
  702. cp_prop_name);
  703. if (cp_sup_cnt < 0)
  704. cp_sup_cnt = 0;
  705. num_supplies = static_sup_cnt + ond_sup_cnt + cp_sup_cnt;
  706. if (num_supplies <= 0) {
  707. dev_err_ratelimited(dev, "%s: supply count is 0 or negative\n", __func__);
  708. rc = -EINVAL;
  709. goto err_supply_cnt;
  710. }
  711. cdc_reg = devm_kcalloc(dev, num_supplies,
  712. sizeof(struct cdc_regulator),
  713. GFP_KERNEL);
  714. if (!cdc_reg) {
  715. rc = -ENOMEM;
  716. goto err_mem_alloc;
  717. }
  718. rc = msm_cdc_parse_supplies(dev, cdc_reg, static_prop_name,
  719. static_sup_cnt, false);
  720. if (rc) {
  721. dev_err_ratelimited(dev, "%s: failed to parse static supplies(%d)\n",
  722. __func__, rc);
  723. goto err_sup;
  724. }
  725. rc = msm_cdc_parse_supplies(dev, &cdc_reg[static_sup_cnt],
  726. ond_prop_name, ond_sup_cnt,
  727. true);
  728. if (rc) {
  729. dev_err_ratelimited(dev, "%s: failed to parse demand supplies(%d)\n",
  730. __func__, rc);
  731. goto err_sup;
  732. }
  733. rc = msm_cdc_parse_supplies(dev,
  734. &cdc_reg[static_sup_cnt + ond_sup_cnt],
  735. cp_prop_name, cp_sup_cnt, true);
  736. if (rc) {
  737. dev_err_ratelimited(dev, "%s: failed to parse cp supplies(%d)\n",
  738. __func__, rc);
  739. goto err_sup;
  740. }
  741. *cdc_vreg = cdc_reg;
  742. *total_num_supplies = num_supplies;
  743. return 0;
  744. err_sup:
  745. err_supply_cnt:
  746. err_mem_alloc:
  747. return rc;
  748. }
  749. EXPORT_SYMBOL(msm_cdc_get_power_supplies);
  750. /*
  751. * msm_cdc_init_wcd_supply:
  752. * Initialize wcd supply parameters.
  753. *
  754. * @np: device node pointer to codec device
  755. * @name: power supply name
  756. * @cdc_supply: codec supply struct to hold wcd params
  757. *
  758. * Return error code if init failed
  759. */
  760. int msm_cdc_init_wcd_supply(struct device_node *np, const char *name,
  761. struct cdc_wcd_supply *cdc_supply)
  762. {
  763. struct platform_device *pdev = NULL;
  764. if (!np || !cdc_supply)
  765. return -EINVAL;
  766. pdev = of_find_device_by_node(np);
  767. if (!pdev)
  768. return -EINVAL;
  769. cdc_supply->dev = &pdev->dev;
  770. cdc_supply->name = name;
  771. cdc_supply->component = snd_soc_lookup_component(&pdev->dev, NULL);
  772. return 0;
  773. }
  774. EXPORT_SYMBOL(msm_cdc_init_wcd_supply);
  775. /*
  776. * msm_cdc_enable_wcd_supply:
  777. * Enable/Disable wcd supply.
  778. *
  779. * @cdc_supply: codec supply struct to hold wcd params
  780. * @enable: bool to inform whether to enable or disable
  781. *
  782. * Return error code if enable/disable failed
  783. */
  784. int msm_cdc_enable_wcd_supply(struct cdc_wcd_supply *cdc_supply, bool enable)
  785. {
  786. struct snd_soc_component *component = cdc_supply->component;
  787. int rc;
  788. if (!component) {
  789. pr_err_ratelimited("%s: Component memory is NULL\n", __func__);
  790. return -EINVAL;
  791. }
  792. if (enable)
  793. rc = snd_soc_dapm_force_enable_pin(
  794. snd_soc_component_get_dapm(component),
  795. cdc_supply->name);
  796. else
  797. rc = snd_soc_dapm_disable_pin(
  798. snd_soc_component_get_dapm(component),
  799. cdc_supply->name);
  800. if (!rc)
  801. snd_soc_dapm_sync(snd_soc_component_get_dapm(component));
  802. else
  803. dev_err_ratelimited(component->dev, "%s: micbias %s force %s pin failed\n",
  804. __func__, cdc_supply->name, (enable ? "enable" : "disable"));
  805. return rc;
  806. }
  807. EXPORT_SYMBOL(msm_cdc_enable_wcd_supply);