power.c 23 KB

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