power.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2015-2021, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/clk.h>
  6. #include <linux/delay.h>
  7. #if IS_ENABLED(CONFIG_MSM_QMP)
  8. #include <linux/mailbox/qmp.h>
  9. #endif
  10. #include <linux/of.h>
  11. #include <linux/pinctrl/consumer.h>
  12. #include <linux/regulator/consumer.h>
  13. #include <soc/qcom/cmd-db.h>
  14. #include "main.h"
  15. #include "qmi.h"
  16. #include "debug.h"
  17. #include "power.h"
  18. static struct icnss_vreg_cfg icnss_wcn6750_vreg_list[] = {
  19. {"vdd-cx-mx", 824000, 952000, 0, 0, 0, false, true},
  20. {"vdd-1.8-xo", 1872000, 1872000, 0, 0, 0, false, true},
  21. {"vdd-1.3-rfa", 1256000, 1352000, 0, 0, 0, false, true},
  22. };
  23. static struct icnss_vreg_cfg icnss_adrestea_vreg_list[] = {
  24. {"vdd-cx-mx", 752000, 752000, 0, 0, 0, false, true},
  25. {"vdd-1.8-xo", 1800000, 1800000, 0, 0, 0, false, true},
  26. {"vdd-1.3-rfa", 1304000, 1304000, 0, 0, 0, false, true},
  27. {"vdd-3.3-ch1", 3312000, 3312000, 0, 0, 0, false, true},
  28. {"vdd-3.3-ch0", 3312000, 3312000, 0, 0, 0, false, true},
  29. };
  30. static struct icnss_battery_level icnss_battery_level[] = {
  31. {70, 3300000},
  32. {60, 3200000},
  33. {50, 3100000},
  34. {25, 3000000},
  35. {0, 2850000},
  36. };
  37. static struct icnss_clk_cfg icnss_clk_list[] = {
  38. {"rf_clk", 0, 0},
  39. };
  40. static struct icnss_clk_cfg icnss_adrestea_clk_list[] = {
  41. {"cxo_ref_clk_pin", 0, 0},
  42. };
  43. #define ICNSS_VREG_LIST_SIZE ARRAY_SIZE(icnss_wcn6750_vreg_list)
  44. #define ICNSS_VREG_ADRESTEA_LIST_SIZE ARRAY_SIZE(icnss_adrestea_vreg_list)
  45. #define ICNSS_CLK_LIST_SIZE ARRAY_SIZE(icnss_clk_list)
  46. #define ICNSS_CLK_ADRESTEA_LIST_SIZE ARRAY_SIZE(icnss_adrestea_clk_list)
  47. #define ICNSS_CHAIN1_REGULATOR "vdd-3.3-ch1"
  48. #define MAX_PROP_SIZE 32
  49. #define BT_CXMX_VOLTAGE_MV 950
  50. #define ICNSS_MBOX_MSG_MAX_LEN 64
  51. #define ICNSS_MBOX_TIMEOUT_MS 1000
  52. #define ICNSS_BATTERY_LEVEL_COUNT ARRAY_SIZE(icnss_battery_level)
  53. #define ICNSS_MAX_BATTERY_LEVEL 100
  54. /**
  55. * enum icnss_vreg_param: Voltage regulator TCS param
  56. * @ICNSS_VREG_VOLTAGE: Provides voltage level to be configured in TCS
  57. * @ICNSS_VREG_MODE: Regulator mode
  58. * @ICNSS_VREG_ENABLE: Set Voltage regulator enable config in TCS
  59. */
  60. enum icnss_vreg_param {
  61. ICNSS_VREG_VOLTAGE,
  62. ICNSS_VREG_MODE,
  63. ICNSS_VREG_ENABLE,
  64. };
  65. /**
  66. * enum icnss_tcs_seq: TCS sequence ID for trigger
  67. * ICNSS_TCS_UP_SEQ: TCS Sequence based on up trigger / Wake TCS
  68. * ICNSS_TCS_DOWN_SEQ: TCS Sequence based on down trigger / Sleep TCS
  69. * ICNSS_TCS_ALL_SEQ: Update for both up and down triggers
  70. */
  71. enum icnss_tcs_seq {
  72. ICNSS_TCS_UP_SEQ,
  73. ICNSS_TCS_DOWN_SEQ,
  74. ICNSS_TCS_ALL_SEQ,
  75. };
  76. static int icnss_get_vreg_single(struct icnss_priv *priv,
  77. struct icnss_vreg_info *vreg)
  78. {
  79. int ret = 0;
  80. struct device *dev = NULL;
  81. struct regulator *reg = NULL;
  82. const __be32 *prop = NULL;
  83. char prop_name[MAX_PROP_SIZE] = {0};
  84. int len = 0;
  85. int i;
  86. dev = &priv->pdev->dev;
  87. reg = devm_regulator_get_optional(dev, vreg->cfg.name);
  88. if (IS_ERR(reg)) {
  89. ret = PTR_ERR(reg);
  90. if (ret == -ENODEV) {
  91. return ret;
  92. } else if (ret == -EPROBE_DEFER) {
  93. icnss_pr_info("EPROBE_DEFER for regulator: %s\n",
  94. vreg->cfg.name);
  95. goto out;
  96. } else if (priv->device_id == ADRASTEA_DEVICE_ID) {
  97. if (vreg->cfg.required) {
  98. icnss_pr_err("Regulator %s doesn't exist: %d\n",
  99. vreg->cfg.name, ret);
  100. goto out;
  101. } else {
  102. icnss_pr_dbg("Optional regulator %s doesn't exist: %d\n",
  103. vreg->cfg.name, ret);
  104. goto done;
  105. }
  106. } else {
  107. icnss_pr_err("Failed to get regulator %s, err = %d\n",
  108. vreg->cfg.name, ret);
  109. goto out;
  110. }
  111. }
  112. vreg->reg = reg;
  113. snprintf(prop_name, MAX_PROP_SIZE, "qcom,%s-config",
  114. vreg->cfg.name);
  115. prop = of_get_property(dev->of_node, prop_name, &len);
  116. icnss_pr_dbg("Got regulator config, prop: %s, len: %d\n",
  117. prop_name, len);
  118. if (!prop || len < (2 * sizeof(__be32))) {
  119. icnss_pr_dbg("Property %s %s, use default\n", prop_name,
  120. prop ? "invalid format" : "doesn't exist");
  121. goto done;
  122. }
  123. for (i = 0; (i * sizeof(__be32)) < len; i++) {
  124. switch (i) {
  125. case 0:
  126. vreg->cfg.min_uv = be32_to_cpup(&prop[0]);
  127. break;
  128. case 1:
  129. vreg->cfg.max_uv = be32_to_cpup(&prop[1]);
  130. break;
  131. case 2:
  132. vreg->cfg.load_ua = be32_to_cpup(&prop[2]);
  133. break;
  134. case 3:
  135. vreg->cfg.delay_us = be32_to_cpup(&prop[3]);
  136. break;
  137. case 4:
  138. if (priv->device_id == WCN6750_DEVICE_ID)
  139. vreg->cfg.need_unvote = be32_to_cpup(&prop[4]);
  140. else
  141. vreg->cfg.need_unvote = 0;
  142. break;
  143. default:
  144. icnss_pr_dbg("Property %s, ignoring value at %d\n",
  145. prop_name, i);
  146. break;
  147. }
  148. }
  149. done:
  150. icnss_pr_dbg("Got regulator: %s, min_uv: %u, max_uv: %u, load_ua: %u, delay_us: %u, need_unvote: %u\n",
  151. vreg->cfg.name, vreg->cfg.min_uv,
  152. vreg->cfg.max_uv, vreg->cfg.load_ua,
  153. vreg->cfg.delay_us, vreg->cfg.need_unvote);
  154. return 0;
  155. out:
  156. return ret;
  157. }
  158. static int icnss_vreg_on_single(struct icnss_vreg_info *vreg)
  159. {
  160. int ret = 0;
  161. if (vreg->enabled) {
  162. icnss_pr_dbg("Regulator %s is already enabled\n",
  163. vreg->cfg.name);
  164. return 0;
  165. }
  166. icnss_pr_dbg("Regulator %s is being enabled\n", vreg->cfg.name);
  167. if (vreg->cfg.min_uv != 0 && vreg->cfg.max_uv != 0) {
  168. ret = regulator_set_voltage(vreg->reg,
  169. vreg->cfg.min_uv,
  170. vreg->cfg.max_uv);
  171. if (ret) {
  172. icnss_pr_err("Failed to set voltage for regulator %s, min_uv: %u, max_uv: %u, err = %d\n",
  173. vreg->cfg.name, vreg->cfg.min_uv,
  174. vreg->cfg.max_uv, ret);
  175. goto out;
  176. }
  177. }
  178. if (vreg->cfg.load_ua) {
  179. ret = regulator_set_load(vreg->reg,
  180. vreg->cfg.load_ua);
  181. if (ret < 0) {
  182. icnss_pr_err("Failed to set load for regulator %s, load: %u, err = %d\n",
  183. vreg->cfg.name, vreg->cfg.load_ua,
  184. ret);
  185. goto out;
  186. }
  187. }
  188. if (vreg->cfg.delay_us)
  189. udelay(vreg->cfg.delay_us);
  190. ret = regulator_enable(vreg->reg);
  191. if (ret) {
  192. icnss_pr_err("Failed to enable regulator %s, err = %d\n",
  193. vreg->cfg.name, ret);
  194. goto out;
  195. }
  196. vreg->enabled = true;
  197. out:
  198. return ret;
  199. }
  200. static int icnss_vreg_unvote_single(struct icnss_vreg_info *vreg)
  201. {
  202. int ret = 0;
  203. if (!vreg->enabled) {
  204. icnss_pr_dbg("Regulator %s is already disabled\n",
  205. vreg->cfg.name);
  206. return 0;
  207. }
  208. icnss_pr_dbg("Removing vote for Regulator %s\n", vreg->cfg.name);
  209. if (vreg->cfg.load_ua) {
  210. ret = regulator_set_load(vreg->reg, 0);
  211. if (ret < 0)
  212. icnss_pr_err("Failed to set load for regulator %s, err = %d\n",
  213. vreg->cfg.name, ret);
  214. }
  215. if (vreg->cfg.min_uv != 0 && vreg->cfg.max_uv != 0) {
  216. ret = regulator_set_voltage(vreg->reg, 0,
  217. vreg->cfg.max_uv);
  218. if (ret)
  219. icnss_pr_err("Failed to set voltage for regulator %s, err = %d\n",
  220. vreg->cfg.name, ret);
  221. }
  222. return ret;
  223. }
  224. static int icnss_vreg_off_single(struct icnss_vreg_info *vreg)
  225. {
  226. int ret = 0;
  227. if (!vreg->enabled) {
  228. icnss_pr_dbg("Regulator %s is already disabled\n",
  229. vreg->cfg.name);
  230. return 0;
  231. }
  232. icnss_pr_dbg("Regulator %s is being disabled\n",
  233. vreg->cfg.name);
  234. ret = regulator_disable(vreg->reg);
  235. if (ret)
  236. icnss_pr_err("Failed to disable regulator %s, err = %d\n",
  237. vreg->cfg.name, ret);
  238. if (vreg->cfg.load_ua) {
  239. ret = regulator_set_load(vreg->reg, 0);
  240. if (ret < 0)
  241. icnss_pr_err("Failed to set load for regulator %s, err = %d\n",
  242. vreg->cfg.name, ret);
  243. }
  244. if (vreg->cfg.min_uv != 0 && vreg->cfg.max_uv != 0) {
  245. ret = regulator_set_voltage(vreg->reg, 0,
  246. vreg->cfg.max_uv);
  247. if (ret)
  248. icnss_pr_err("Failed to set voltage for regulator %s, err = %d\n",
  249. vreg->cfg.name, ret);
  250. }
  251. vreg->enabled = false;
  252. return ret;
  253. }
  254. static struct icnss_vreg_cfg *get_vreg_list(u32 *vreg_list_size,
  255. unsigned long device_id)
  256. {
  257. switch (device_id) {
  258. case WCN6750_DEVICE_ID:
  259. *vreg_list_size = ICNSS_VREG_LIST_SIZE;
  260. return icnss_wcn6750_vreg_list;
  261. case ADRASTEA_DEVICE_ID:
  262. *vreg_list_size = ICNSS_VREG_ADRESTEA_LIST_SIZE;
  263. return icnss_adrestea_vreg_list;
  264. default:
  265. icnss_pr_err("Unsupported device_id 0x%x\n", device_id);
  266. *vreg_list_size = 0;
  267. return NULL;
  268. }
  269. }
  270. int icnss_get_vreg(struct icnss_priv *priv)
  271. {
  272. int ret = 0;
  273. int i;
  274. struct icnss_vreg_info *vreg;
  275. struct icnss_vreg_cfg *vreg_cfg = NULL;
  276. struct list_head *vreg_list = &priv->vreg_list;
  277. struct device *dev = &priv->pdev->dev;
  278. u32 vreg_list_size = 0;
  279. vreg_cfg = get_vreg_list(&vreg_list_size, priv->device_id);
  280. if (!vreg_cfg)
  281. return -EINVAL;
  282. for (i = 0; i < vreg_list_size; i++) {
  283. vreg = devm_kzalloc(dev, sizeof(*vreg), GFP_KERNEL);
  284. if (!vreg)
  285. return -ENOMEM;
  286. memcpy(&vreg->cfg, &vreg_cfg[i], sizeof(vreg->cfg));
  287. ret = icnss_get_vreg_single(priv, vreg);
  288. if (ret != 0) {
  289. if (ret == -ENODEV)
  290. continue;
  291. else
  292. return ret;
  293. }
  294. list_add_tail(&vreg->list, vreg_list);
  295. }
  296. return 0;
  297. }
  298. void icnss_put_vreg(struct icnss_priv *priv)
  299. {
  300. struct list_head *vreg_list = &priv->vreg_list;
  301. struct icnss_vreg_info *vreg = NULL;
  302. while (!list_empty(vreg_list)) {
  303. vreg = list_first_entry(vreg_list,
  304. struct icnss_vreg_info, list);
  305. list_del(&vreg->list);
  306. }
  307. }
  308. static int icnss_vreg_on(struct icnss_priv *priv)
  309. {
  310. struct list_head *vreg_list = &priv->vreg_list;
  311. struct icnss_vreg_info *vreg = NULL;
  312. int ret = 0;
  313. list_for_each_entry(vreg, vreg_list, list) {
  314. if (IS_ERR_OR_NULL(vreg->reg) || !vreg->cfg.is_supported)
  315. continue;
  316. if (!priv->chain_reg_info_updated &&
  317. !strcmp(ICNSS_CHAIN1_REGULATOR, vreg->cfg.name)) {
  318. priv->chain_reg_info_updated = true;
  319. if (!priv->is_chain1_supported) {
  320. vreg->cfg.is_supported = false;
  321. continue;
  322. }
  323. }
  324. ret = icnss_vreg_on_single(vreg);
  325. if (ret)
  326. break;
  327. }
  328. if (!ret)
  329. return 0;
  330. list_for_each_entry_continue_reverse(vreg, vreg_list, list) {
  331. if (IS_ERR_OR_NULL(vreg->reg) || !vreg->enabled)
  332. continue;
  333. icnss_vreg_off_single(vreg);
  334. }
  335. return ret;
  336. }
  337. static int icnss_vreg_off(struct icnss_priv *priv)
  338. {
  339. struct list_head *vreg_list = &priv->vreg_list;
  340. struct icnss_vreg_info *vreg = NULL;
  341. list_for_each_entry_reverse(vreg, vreg_list, list) {
  342. if (IS_ERR_OR_NULL(vreg->reg))
  343. continue;
  344. icnss_vreg_off_single(vreg);
  345. }
  346. return 0;
  347. }
  348. int icnss_vreg_unvote(struct icnss_priv *priv)
  349. {
  350. struct list_head *vreg_list = &priv->vreg_list;
  351. struct icnss_vreg_info *vreg = NULL;
  352. list_for_each_entry_reverse(vreg, vreg_list, list) {
  353. if (IS_ERR_OR_NULL(vreg->reg))
  354. continue;
  355. if (vreg->cfg.need_unvote)
  356. icnss_vreg_unvote_single(vreg);
  357. }
  358. return 0;
  359. }
  360. int icnss_get_clk_single(struct icnss_priv *priv,
  361. struct icnss_clk_info *clk_info)
  362. {
  363. struct device *dev = &priv->pdev->dev;
  364. struct clk *clk;
  365. int ret;
  366. clk = devm_clk_get(dev, clk_info->cfg.name);
  367. if (IS_ERR(clk)) {
  368. ret = PTR_ERR(clk);
  369. if (clk_info->cfg.required)
  370. icnss_pr_err("Failed to get clock %s, err = %d\n",
  371. clk_info->cfg.name, ret);
  372. else
  373. icnss_pr_dbg("Failed to get optional clock %s, err = %d\n",
  374. clk_info->cfg.name, ret);
  375. return ret;
  376. }
  377. clk_info->clk = clk;
  378. icnss_pr_dbg("Got clock: %s, freq: %u\n",
  379. clk_info->cfg.name, clk_info->cfg.freq);
  380. return 0;
  381. }
  382. static int icnss_clk_on_single(struct icnss_clk_info *clk_info)
  383. {
  384. int ret;
  385. if (clk_info->enabled) {
  386. icnss_pr_dbg("Clock %s is already enabled\n",
  387. clk_info->cfg.name);
  388. return 0;
  389. }
  390. icnss_pr_dbg("Clock %s is being enabled\n", clk_info->cfg.name);
  391. if (clk_info->cfg.freq) {
  392. ret = clk_set_rate(clk_info->clk, clk_info->cfg.freq);
  393. if (ret) {
  394. icnss_pr_err("Failed to set frequency %u for clock %s, err = %d\n",
  395. clk_info->cfg.freq, clk_info->cfg.name,
  396. ret);
  397. return ret;
  398. }
  399. }
  400. ret = clk_prepare_enable(clk_info->clk);
  401. if (ret) {
  402. icnss_pr_err("Failed to enable clock %s, err = %d\n",
  403. clk_info->cfg.name, ret);
  404. return ret;
  405. }
  406. clk_info->enabled = true;
  407. return 0;
  408. }
  409. static int icnss_clk_off_single(struct icnss_clk_info *clk_info)
  410. {
  411. if (!clk_info->enabled) {
  412. icnss_pr_dbg("Clock %s is already disabled\n",
  413. clk_info->cfg.name);
  414. return 0;
  415. }
  416. icnss_pr_dbg("Clock %s is being disabled\n", clk_info->cfg.name);
  417. clk_disable_unprepare(clk_info->clk);
  418. clk_info->enabled = false;
  419. return 0;
  420. }
  421. int icnss_get_clk(struct icnss_priv *priv)
  422. {
  423. struct device *dev;
  424. struct list_head *clk_list;
  425. struct icnss_clk_info *clk_info;
  426. struct icnss_clk_cfg *clk_cfg;
  427. int ret, i;
  428. u32 clk_list_size = 0;
  429. if (!priv)
  430. return -ENODEV;
  431. dev = &priv->pdev->dev;
  432. clk_list = &priv->clk_list;
  433. if (priv->device_id == ADRASTEA_DEVICE_ID) {
  434. clk_cfg = icnss_adrestea_clk_list;
  435. clk_list_size = ICNSS_CLK_ADRESTEA_LIST_SIZE;
  436. } else if (priv->device_id == WCN6750_DEVICE_ID) {
  437. clk_cfg = icnss_clk_list;
  438. clk_list_size = ICNSS_CLK_LIST_SIZE;
  439. }
  440. if (!list_empty(clk_list)) {
  441. icnss_pr_dbg("Clocks have already been updated\n");
  442. return 0;
  443. }
  444. for (i = 0; i < clk_list_size; i++) {
  445. clk_info = devm_kzalloc(dev, sizeof(*clk_info), GFP_KERNEL);
  446. if (!clk_info) {
  447. ret = -ENOMEM;
  448. goto cleanup;
  449. }
  450. memcpy(&clk_info->cfg, &clk_cfg[i],
  451. sizeof(clk_info->cfg));
  452. ret = icnss_get_clk_single(priv, clk_info);
  453. if (ret != 0) {
  454. if (clk_info->cfg.required)
  455. goto cleanup;
  456. else
  457. continue;
  458. }
  459. list_add_tail(&clk_info->list, clk_list);
  460. }
  461. return 0;
  462. cleanup:
  463. while (!list_empty(clk_list)) {
  464. clk_info = list_first_entry(clk_list, struct icnss_clk_info,
  465. list);
  466. list_del(&clk_info->list);
  467. }
  468. return ret;
  469. }
  470. void icnss_put_clk(struct icnss_priv *priv)
  471. {
  472. struct device *dev;
  473. struct list_head *clk_list;
  474. struct icnss_clk_info *clk_info;
  475. if (!priv)
  476. return;
  477. dev = &priv->pdev->dev;
  478. clk_list = &priv->clk_list;
  479. while (!list_empty(clk_list)) {
  480. clk_info = list_first_entry(clk_list, struct icnss_clk_info,
  481. list);
  482. list_del(&clk_info->list);
  483. }
  484. }
  485. static int icnss_clk_on(struct list_head *clk_list)
  486. {
  487. struct icnss_clk_info *clk_info;
  488. int ret = 0;
  489. list_for_each_entry(clk_info, clk_list, list) {
  490. if (IS_ERR_OR_NULL(clk_info->clk))
  491. continue;
  492. ret = icnss_clk_on_single(clk_info);
  493. if (ret)
  494. break;
  495. }
  496. if (!ret)
  497. return 0;
  498. list_for_each_entry_continue_reverse(clk_info, clk_list, list) {
  499. if (IS_ERR_OR_NULL(clk_info->clk))
  500. continue;
  501. icnss_clk_off_single(clk_info);
  502. }
  503. return ret;
  504. }
  505. static int icnss_clk_off(struct list_head *clk_list)
  506. {
  507. struct icnss_clk_info *clk_info;
  508. list_for_each_entry_reverse(clk_info, clk_list, list) {
  509. if (IS_ERR_OR_NULL(clk_info->clk))
  510. continue;
  511. icnss_clk_off_single(clk_info);
  512. }
  513. return 0;
  514. }
  515. int icnss_hw_power_on(struct icnss_priv *priv)
  516. {
  517. int ret = 0;
  518. icnss_pr_dbg("HW Power on: state: 0x%lx\n", priv->state);
  519. spin_lock(&priv->on_off_lock);
  520. if (test_bit(ICNSS_POWER_ON, &priv->state)) {
  521. spin_unlock(&priv->on_off_lock);
  522. return ret;
  523. }
  524. set_bit(ICNSS_POWER_ON, &priv->state);
  525. spin_unlock(&priv->on_off_lock);
  526. ret = icnss_vreg_on(priv);
  527. if (ret) {
  528. icnss_pr_err("Failed to turn on vreg, err = %d\n", ret);
  529. goto out;
  530. }
  531. ret = icnss_clk_on(&priv->clk_list);
  532. if (ret)
  533. goto vreg_off;
  534. return ret;
  535. vreg_off:
  536. icnss_vreg_off(priv);
  537. out:
  538. clear_bit(ICNSS_POWER_ON, &priv->state);
  539. return ret;
  540. }
  541. int icnss_hw_power_off(struct icnss_priv *priv)
  542. {
  543. int ret = 0;
  544. if (test_bit(HW_ALWAYS_ON, &priv->ctrl_params.quirks))
  545. return 0;
  546. if (test_bit(ICNSS_FW_DOWN, &priv->state))
  547. return 0;
  548. icnss_pr_dbg("HW Power off: 0x%lx\n", priv->state);
  549. spin_lock(&priv->on_off_lock);
  550. if (!test_bit(ICNSS_POWER_ON, &priv->state)) {
  551. spin_unlock(&priv->on_off_lock);
  552. return ret;
  553. }
  554. clear_bit(ICNSS_POWER_ON, &priv->state);
  555. spin_unlock(&priv->on_off_lock);
  556. icnss_clk_off(&priv->clk_list);
  557. ret = icnss_vreg_off(priv);
  558. return ret;
  559. }
  560. int icnss_power_on(struct device *dev)
  561. {
  562. struct icnss_priv *priv = dev_get_drvdata(dev);
  563. if (!priv) {
  564. icnss_pr_err("Invalid drvdata: dev %pK, data %pK\n",
  565. dev, priv);
  566. return -EINVAL;
  567. }
  568. icnss_pr_dbg("Power On: 0x%lx\n", priv->state);
  569. return icnss_hw_power_on(priv);
  570. }
  571. EXPORT_SYMBOL(icnss_power_on);
  572. int icnss_power_off(struct device *dev)
  573. {
  574. struct icnss_priv *priv = dev_get_drvdata(dev);
  575. if (!priv) {
  576. icnss_pr_err("Invalid drvdata: dev %pK, data %pK\n",
  577. dev, priv);
  578. return -EINVAL;
  579. }
  580. icnss_pr_dbg("Power Off: 0x%lx\n", priv->state);
  581. return icnss_hw_power_off(priv);
  582. }
  583. EXPORT_SYMBOL(icnss_power_off);
  584. void icnss_put_resources(struct icnss_priv *priv)
  585. {
  586. icnss_put_clk(priv);
  587. icnss_put_vreg(priv);
  588. }
  589. int icnss_aop_mbox_init(struct icnss_priv *priv)
  590. {
  591. struct mbox_client *mbox = &priv->mbox_client_data;
  592. struct mbox_chan *chan;
  593. int ret = 0;
  594. ret = of_property_read_string(priv->pdev->dev.of_node,
  595. "qcom,vreg_ol_cpr",
  596. &priv->cpr_info.vreg_ol_cpr);
  597. if (ret) {
  598. icnss_pr_dbg("Vreg for OL CPR not configured\n");
  599. return -EINVAL;
  600. }
  601. mbox->dev = &priv->pdev->dev;
  602. mbox->tx_block = true;
  603. mbox->tx_tout = ICNSS_MBOX_TIMEOUT_MS;
  604. mbox->knows_txdone = false;
  605. priv->mbox_chan = NULL;
  606. chan = mbox_request_channel(mbox, 0);
  607. if (IS_ERR(chan)) {
  608. ret = PTR_ERR(chan);
  609. icnss_pr_err("Failed to get mbox channel with err %d\n", ret);
  610. return ret;
  611. }
  612. priv->mbox_chan = chan;
  613. icnss_pr_dbg("Mbox channel initialized\n");
  614. return 0;
  615. }
  616. #if IS_ENABLED(CONFIG_MSM_QMP)
  617. static int icnss_aop_set_vreg_param(struct icnss_priv *priv,
  618. const char *vreg_name,
  619. enum icnss_vreg_param param,
  620. enum icnss_tcs_seq seq, int val)
  621. {
  622. struct qmp_pkt pkt;
  623. char mbox_msg[ICNSS_MBOX_MSG_MAX_LEN];
  624. static const char * const vreg_param_str[] = {"v", "m", "e"};
  625. static const char *const tcs_seq_str[] = {"upval", "dwnval", "enable"};
  626. int ret = 0;
  627. if (param > ICNSS_VREG_ENABLE || seq > ICNSS_TCS_ALL_SEQ || !vreg_name)
  628. return -EINVAL;
  629. snprintf(mbox_msg, ICNSS_MBOX_MSG_MAX_LEN,
  630. "{class: wlan_pdc, res: %s.%s, %s: %d}", vreg_name,
  631. vreg_param_str[param], tcs_seq_str[seq], val);
  632. icnss_pr_dbg("Sending AOP Mbox msg: %s\n", mbox_msg);
  633. pkt.size = ICNSS_MBOX_MSG_MAX_LEN;
  634. pkt.data = mbox_msg;
  635. ret = mbox_send_message(priv->mbox_chan, &pkt);
  636. if (ret < 0)
  637. icnss_pr_err("Failed to send AOP mbox msg: %s,ret: %d\n",
  638. mbox_msg, ret);
  639. else
  640. ret = 0;
  641. return ret;
  642. }
  643. #else
  644. static int icnss_aop_set_vreg_param(struct icnss_priv *priv,
  645. const char *vreg_name,
  646. enum icnss_vreg_param param,
  647. enum icnss_tcs_seq seq, int val)
  648. {
  649. return 0;
  650. }
  651. #endif
  652. int icnss_update_cpr_info(struct icnss_priv *priv)
  653. {
  654. struct icnss_cpr_info *cpr_info = &priv->cpr_info;
  655. if (!cpr_info->vreg_ol_cpr || !priv->mbox_chan) {
  656. icnss_pr_dbg("Mbox channel / OL CPR Vreg not configured\n");
  657. return 0;
  658. }
  659. if (cpr_info->voltage == 0) {
  660. icnss_pr_err("Voltage %dmV is not valid\n", cpr_info->voltage);
  661. return -EINVAL;
  662. }
  663. cpr_info->voltage = cpr_info->voltage > BT_CXMX_VOLTAGE_MV ?
  664. cpr_info->voltage : BT_CXMX_VOLTAGE_MV;
  665. return icnss_aop_set_vreg_param(priv,
  666. cpr_info->vreg_ol_cpr,
  667. ICNSS_VREG_VOLTAGE,
  668. ICNSS_TCS_UP_SEQ,
  669. cpr_info->voltage);
  670. }
  671. static int icnss_get_battery_level(struct icnss_priv *priv)
  672. {
  673. int err = 0, battery_percentage = 0;
  674. union power_supply_propval psp = {0,};
  675. if (!priv->batt_psy)
  676. priv->batt_psy = power_supply_get_by_name("battery");
  677. if (priv->batt_psy) {
  678. err = power_supply_get_property(priv->batt_psy,
  679. POWER_SUPPLY_PROP_CAPACITY,
  680. &psp);
  681. if (err) {
  682. icnss_pr_err("battery percentage read error:%d\n", err);
  683. goto out;
  684. }
  685. battery_percentage = psp.intval;
  686. }
  687. icnss_pr_info("Battery Percentage: %d\n", battery_percentage);
  688. out:
  689. return battery_percentage;
  690. }
  691. static void icnss_update_soc_level(struct work_struct *work)
  692. {
  693. int battery_percentage = 0, current_updated_voltage = 0, err = 0;
  694. int level_count;
  695. struct icnss_priv *priv = container_of(work, struct icnss_priv, soc_update_work);
  696. battery_percentage = icnss_get_battery_level(priv);
  697. if (!battery_percentage ||
  698. battery_percentage > ICNSS_MAX_BATTERY_LEVEL) {
  699. icnss_pr_err("Battery percentage read failure\n");
  700. return;
  701. }
  702. for (level_count = 0; level_count < ICNSS_BATTERY_LEVEL_COUNT;
  703. level_count++) {
  704. if (battery_percentage >=
  705. icnss_battery_level[level_count].lower_battery_threshold) {
  706. current_updated_voltage =
  707. icnss_battery_level[level_count].ldo_voltage;
  708. break;
  709. }
  710. }
  711. if (level_count != ICNSS_BATTERY_LEVEL_COUNT &&
  712. priv->last_updated_voltage != current_updated_voltage) {
  713. err = icnss_send_vbatt_update(priv, current_updated_voltage);
  714. if (err < 0) {
  715. icnss_pr_err("Unable to update ldo voltage");
  716. return;
  717. }
  718. priv->last_updated_voltage = current_updated_voltage;
  719. }
  720. }
  721. static int icnss_battery_supply_callback(struct notifier_block *nb,
  722. unsigned long event, void *data)
  723. {
  724. struct power_supply *psy = data;
  725. struct icnss_priv *priv = container_of(nb, struct icnss_priv,
  726. psf_nb);
  727. if (strcmp(psy->desc->name, "battery"))
  728. return NOTIFY_OK;
  729. if (test_bit(ICNSS_WLFW_CONNECTED, &priv->state) &&
  730. !test_bit(ICNSS_FW_DOWN, &priv->state))
  731. queue_work(priv->soc_update_wq, &priv->soc_update_work);
  732. return NOTIFY_OK;
  733. }
  734. int icnss_get_psf_info(struct icnss_priv *priv)
  735. {
  736. int ret = 0;
  737. priv->soc_update_wq = alloc_workqueue("icnss_soc_update",
  738. WQ_UNBOUND, 1);
  739. if (!priv->soc_update_wq) {
  740. icnss_pr_err("Workqueue creation failed for soc update\n");
  741. ret = -EFAULT;
  742. goto out;
  743. }
  744. priv->psf_nb.notifier_call = icnss_battery_supply_callback;
  745. ret = power_supply_reg_notifier(&priv->psf_nb);
  746. if (ret < 0) {
  747. icnss_pr_err("Power supply framework registration err: %d\n",
  748. ret);
  749. goto err_psf_registration;
  750. }
  751. INIT_WORK(&priv->soc_update_work, icnss_update_soc_level);
  752. return 0;
  753. err_psf_registration:
  754. destroy_workqueue(priv->soc_update_wq);
  755. out:
  756. return ret;
  757. }