power.c 18 KB

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