88pm886-regulator.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. #include <linux/i2c.h>
  3. #include <linux/module.h>
  4. #include <linux/platform_device.h>
  5. #include <linux/regmap.h>
  6. #include <linux/regulator/driver.h>
  7. #include <linux/mfd/88pm886.h>
  8. static const struct regmap_config pm886_regulator_regmap_config = {
  9. .reg_bits = 8,
  10. .val_bits = 8,
  11. .max_register = PM886_REG_BUCK5_VOUT,
  12. };
  13. static const struct regulator_ops pm886_ldo_ops = {
  14. .list_voltage = regulator_list_voltage_table,
  15. .map_voltage = regulator_map_voltage_iterate,
  16. .set_voltage_sel = regulator_set_voltage_sel_regmap,
  17. .get_voltage_sel = regulator_get_voltage_sel_regmap,
  18. .enable = regulator_enable_regmap,
  19. .disable = regulator_disable_regmap,
  20. .is_enabled = regulator_is_enabled_regmap,
  21. };
  22. static const struct regulator_ops pm886_buck_ops = {
  23. .list_voltage = regulator_list_voltage_linear_range,
  24. .map_voltage = regulator_map_voltage_linear_range,
  25. .set_voltage_sel = regulator_set_voltage_sel_regmap,
  26. .get_voltage_sel = regulator_get_voltage_sel_regmap,
  27. .enable = regulator_enable_regmap,
  28. .disable = regulator_disable_regmap,
  29. .is_enabled = regulator_is_enabled_regmap,
  30. };
  31. static const unsigned int pm886_ldo_volt_table1[] = {
  32. 1700000, 1800000, 1900000, 2500000, 2800000, 2900000, 3100000, 3300000,
  33. };
  34. static const unsigned int pm886_ldo_volt_table2[] = {
  35. 1200000, 1250000, 1700000, 1800000, 1850000, 1900000, 2500000, 2600000,
  36. 2700000, 2750000, 2800000, 2850000, 2900000, 3000000, 3100000, 3300000,
  37. };
  38. static const unsigned int pm886_ldo_volt_table3[] = {
  39. 1700000, 1800000, 1900000, 2000000, 2100000, 2500000, 2700000, 2800000,
  40. };
  41. static const struct linear_range pm886_buck_volt_ranges1[] = {
  42. REGULATOR_LINEAR_RANGE(600000, 0, 79, 12500),
  43. REGULATOR_LINEAR_RANGE(1600000, 80, 84, 50000),
  44. };
  45. static const struct linear_range pm886_buck_volt_ranges2[] = {
  46. REGULATOR_LINEAR_RANGE(600000, 0, 79, 12500),
  47. REGULATOR_LINEAR_RANGE(1600000, 80, 114, 50000),
  48. };
  49. static struct regulator_desc pm886_regulators[] = {
  50. {
  51. .name = "LDO1",
  52. .regulators_node = "regulators",
  53. .of_match = "ldo1",
  54. .ops = &pm886_ldo_ops,
  55. .type = REGULATOR_VOLTAGE,
  56. .enable_reg = PM886_REG_LDO_EN1,
  57. .enable_mask = BIT(0),
  58. .volt_table = pm886_ldo_volt_table1,
  59. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table1),
  60. .vsel_reg = PM886_REG_LDO1_VOUT,
  61. .vsel_mask = PM886_LDO_VSEL_MASK,
  62. },
  63. {
  64. .name = "LDO2",
  65. .regulators_node = "regulators",
  66. .of_match = "ldo2",
  67. .ops = &pm886_ldo_ops,
  68. .type = REGULATOR_VOLTAGE,
  69. .enable_reg = PM886_REG_LDO_EN1,
  70. .enable_mask = BIT(1),
  71. .volt_table = pm886_ldo_volt_table1,
  72. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table1),
  73. .vsel_reg = PM886_REG_LDO2_VOUT,
  74. .vsel_mask = PM886_LDO_VSEL_MASK,
  75. },
  76. {
  77. .name = "LDO3",
  78. .regulators_node = "regulators",
  79. .of_match = "ldo3",
  80. .ops = &pm886_ldo_ops,
  81. .type = REGULATOR_VOLTAGE,
  82. .enable_reg = PM886_REG_LDO_EN1,
  83. .enable_mask = BIT(2),
  84. .volt_table = pm886_ldo_volt_table1,
  85. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table1),
  86. .vsel_reg = PM886_REG_LDO3_VOUT,
  87. .vsel_mask = PM886_LDO_VSEL_MASK,
  88. },
  89. {
  90. .name = "LDO4",
  91. .regulators_node = "regulators",
  92. .of_match = "ldo4",
  93. .ops = &pm886_ldo_ops,
  94. .type = REGULATOR_VOLTAGE,
  95. .enable_reg = PM886_REG_LDO_EN1,
  96. .enable_mask = BIT(3),
  97. .volt_table = pm886_ldo_volt_table2,
  98. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  99. .vsel_reg = PM886_REG_LDO4_VOUT,
  100. .vsel_mask = PM886_LDO_VSEL_MASK,
  101. },
  102. {
  103. .name = "LDO5",
  104. .regulators_node = "regulators",
  105. .of_match = "ldo5",
  106. .ops = &pm886_ldo_ops,
  107. .type = REGULATOR_VOLTAGE,
  108. .enable_reg = PM886_REG_LDO_EN1,
  109. .enable_mask = BIT(4),
  110. .volt_table = pm886_ldo_volt_table2,
  111. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  112. .vsel_reg = PM886_REG_LDO5_VOUT,
  113. .vsel_mask = PM886_LDO_VSEL_MASK,
  114. },
  115. {
  116. .name = "LDO6",
  117. .regulators_node = "regulators",
  118. .of_match = "ldo6",
  119. .ops = &pm886_ldo_ops,
  120. .type = REGULATOR_VOLTAGE,
  121. .enable_reg = PM886_REG_LDO_EN1,
  122. .enable_mask = BIT(5),
  123. .volt_table = pm886_ldo_volt_table2,
  124. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  125. .vsel_reg = PM886_REG_LDO6_VOUT,
  126. .vsel_mask = PM886_LDO_VSEL_MASK,
  127. },
  128. {
  129. .name = "LDO7",
  130. .regulators_node = "regulators",
  131. .of_match = "ldo7",
  132. .ops = &pm886_ldo_ops,
  133. .type = REGULATOR_VOLTAGE,
  134. .enable_reg = PM886_REG_LDO_EN1,
  135. .enable_mask = BIT(6),
  136. .volt_table = pm886_ldo_volt_table2,
  137. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  138. .vsel_reg = PM886_REG_LDO7_VOUT,
  139. .vsel_mask = PM886_LDO_VSEL_MASK,
  140. },
  141. {
  142. .name = "LDO8",
  143. .regulators_node = "regulators",
  144. .of_match = "ldo8",
  145. .ops = &pm886_ldo_ops,
  146. .type = REGULATOR_VOLTAGE,
  147. .enable_reg = PM886_REG_LDO_EN1,
  148. .enable_mask = BIT(7),
  149. .volt_table = pm886_ldo_volt_table2,
  150. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  151. .vsel_reg = PM886_REG_LDO8_VOUT,
  152. .vsel_mask = PM886_LDO_VSEL_MASK,
  153. },
  154. {
  155. .name = "LDO9",
  156. .regulators_node = "regulators",
  157. .of_match = "ldo9",
  158. .ops = &pm886_ldo_ops,
  159. .type = REGULATOR_VOLTAGE,
  160. .enable_reg = PM886_REG_LDO_EN2,
  161. .enable_mask = BIT(0),
  162. .volt_table = pm886_ldo_volt_table2,
  163. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  164. .vsel_reg = PM886_REG_LDO9_VOUT,
  165. .vsel_mask = PM886_LDO_VSEL_MASK,
  166. },
  167. {
  168. .name = "LDO10",
  169. .regulators_node = "regulators",
  170. .of_match = "ldo10",
  171. .ops = &pm886_ldo_ops,
  172. .type = REGULATOR_VOLTAGE,
  173. .enable_reg = PM886_REG_LDO_EN2,
  174. .enable_mask = BIT(1),
  175. .volt_table = pm886_ldo_volt_table2,
  176. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  177. .vsel_reg = PM886_REG_LDO10_VOUT,
  178. .vsel_mask = PM886_LDO_VSEL_MASK,
  179. },
  180. {
  181. .name = "LDO11",
  182. .regulators_node = "regulators",
  183. .of_match = "ldo11",
  184. .ops = &pm886_ldo_ops,
  185. .type = REGULATOR_VOLTAGE,
  186. .enable_reg = PM886_REG_LDO_EN2,
  187. .enable_mask = BIT(2),
  188. .volt_table = pm886_ldo_volt_table2,
  189. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  190. .vsel_reg = PM886_REG_LDO11_VOUT,
  191. .vsel_mask = PM886_LDO_VSEL_MASK,
  192. },
  193. {
  194. .name = "LDO12",
  195. .regulators_node = "regulators",
  196. .of_match = "ldo12",
  197. .ops = &pm886_ldo_ops,
  198. .type = REGULATOR_VOLTAGE,
  199. .enable_reg = PM886_REG_LDO_EN2,
  200. .enable_mask = BIT(3),
  201. .volt_table = pm886_ldo_volt_table2,
  202. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  203. .vsel_reg = PM886_REG_LDO12_VOUT,
  204. .vsel_mask = PM886_LDO_VSEL_MASK,
  205. },
  206. {
  207. .name = "LDO13",
  208. .regulators_node = "regulators",
  209. .of_match = "ldo13",
  210. .ops = &pm886_ldo_ops,
  211. .type = REGULATOR_VOLTAGE,
  212. .enable_reg = PM886_REG_LDO_EN2,
  213. .enable_mask = BIT(4),
  214. .volt_table = pm886_ldo_volt_table2,
  215. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  216. .vsel_reg = PM886_REG_LDO13_VOUT,
  217. .vsel_mask = PM886_LDO_VSEL_MASK,
  218. },
  219. {
  220. .name = "LDO14",
  221. .regulators_node = "regulators",
  222. .of_match = "ldo14",
  223. .ops = &pm886_ldo_ops,
  224. .type = REGULATOR_VOLTAGE,
  225. .enable_reg = PM886_REG_LDO_EN2,
  226. .enable_mask = BIT(5),
  227. .volt_table = pm886_ldo_volt_table2,
  228. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  229. .vsel_reg = PM886_REG_LDO14_VOUT,
  230. .vsel_mask = PM886_LDO_VSEL_MASK,
  231. },
  232. {
  233. .name = "LDO15",
  234. .regulators_node = "regulators",
  235. .of_match = "ldo15",
  236. .ops = &pm886_ldo_ops,
  237. .type = REGULATOR_VOLTAGE,
  238. .enable_reg = PM886_REG_LDO_EN2,
  239. .enable_mask = BIT(6),
  240. .volt_table = pm886_ldo_volt_table2,
  241. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table2),
  242. .vsel_reg = PM886_REG_LDO15_VOUT,
  243. .vsel_mask = PM886_LDO_VSEL_MASK,
  244. },
  245. {
  246. .name = "LDO16",
  247. .regulators_node = "regulators",
  248. .of_match = "ldo16",
  249. .ops = &pm886_ldo_ops,
  250. .type = REGULATOR_VOLTAGE,
  251. .enable_reg = PM886_REG_LDO_EN2,
  252. .enable_mask = BIT(7),
  253. .volt_table = pm886_ldo_volt_table3,
  254. .n_voltages = ARRAY_SIZE(pm886_ldo_volt_table3),
  255. .vsel_reg = PM886_REG_LDO16_VOUT,
  256. .vsel_mask = PM886_LDO_VSEL_MASK,
  257. },
  258. {
  259. .name = "buck1",
  260. .regulators_node = "regulators",
  261. .of_match = "buck1",
  262. .ops = &pm886_buck_ops,
  263. .type = REGULATOR_VOLTAGE,
  264. .n_voltages = 85,
  265. .linear_ranges = pm886_buck_volt_ranges1,
  266. .n_linear_ranges = ARRAY_SIZE(pm886_buck_volt_ranges1),
  267. .vsel_reg = PM886_REG_BUCK1_VOUT,
  268. .vsel_mask = PM886_BUCK_VSEL_MASK,
  269. .enable_reg = PM886_REG_BUCK_EN,
  270. .enable_mask = BIT(0),
  271. },
  272. {
  273. .name = "buck2",
  274. .regulators_node = "regulators",
  275. .of_match = "buck2",
  276. .ops = &pm886_buck_ops,
  277. .type = REGULATOR_VOLTAGE,
  278. .n_voltages = 115,
  279. .linear_ranges = pm886_buck_volt_ranges2,
  280. .n_linear_ranges = ARRAY_SIZE(pm886_buck_volt_ranges2),
  281. .vsel_reg = PM886_REG_BUCK2_VOUT,
  282. .vsel_mask = PM886_BUCK_VSEL_MASK,
  283. .enable_reg = PM886_REG_BUCK_EN,
  284. .enable_mask = BIT(1),
  285. },
  286. {
  287. .name = "buck3",
  288. .regulators_node = "regulators",
  289. .of_match = "buck3",
  290. .ops = &pm886_buck_ops,
  291. .type = REGULATOR_VOLTAGE,
  292. .n_voltages = 115,
  293. .linear_ranges = pm886_buck_volt_ranges2,
  294. .n_linear_ranges = ARRAY_SIZE(pm886_buck_volt_ranges2),
  295. .vsel_reg = PM886_REG_BUCK3_VOUT,
  296. .vsel_mask = PM886_BUCK_VSEL_MASK,
  297. .enable_reg = PM886_REG_BUCK_EN,
  298. .enable_mask = BIT(2),
  299. },
  300. {
  301. .name = "buck4",
  302. .regulators_node = "regulators",
  303. .of_match = "buck4",
  304. .ops = &pm886_buck_ops,
  305. .type = REGULATOR_VOLTAGE,
  306. .n_voltages = 115,
  307. .linear_ranges = pm886_buck_volt_ranges2,
  308. .n_linear_ranges = ARRAY_SIZE(pm886_buck_volt_ranges2),
  309. .vsel_reg = PM886_REG_BUCK4_VOUT,
  310. .vsel_mask = PM886_BUCK_VSEL_MASK,
  311. .enable_reg = PM886_REG_BUCK_EN,
  312. .enable_mask = BIT(3),
  313. },
  314. {
  315. .name = "buck5",
  316. .regulators_node = "regulators",
  317. .of_match = "buck5",
  318. .ops = &pm886_buck_ops,
  319. .type = REGULATOR_VOLTAGE,
  320. .n_voltages = 115,
  321. .linear_ranges = pm886_buck_volt_ranges2,
  322. .n_linear_ranges = ARRAY_SIZE(pm886_buck_volt_ranges2),
  323. .vsel_reg = PM886_REG_BUCK5_VOUT,
  324. .vsel_mask = PM886_BUCK_VSEL_MASK,
  325. .enable_reg = PM886_REG_BUCK_EN,
  326. .enable_mask = BIT(4),
  327. },
  328. };
  329. static int pm886_regulator_probe(struct platform_device *pdev)
  330. {
  331. struct pm886_chip *chip = dev_get_drvdata(pdev->dev.parent);
  332. struct regulator_config rcfg = { };
  333. struct device *dev = &pdev->dev;
  334. struct regulator_desc *rdesc;
  335. struct regulator_dev *rdev;
  336. struct i2c_client *page;
  337. struct regmap *regmap;
  338. page = devm_i2c_new_dummy_device(dev, chip->client->adapter,
  339. chip->client->addr + PM886_PAGE_OFFSET_REGULATORS);
  340. if (IS_ERR(page))
  341. return dev_err_probe(dev, PTR_ERR(page),
  342. "Failed to initialize regulators client\n");
  343. regmap = devm_regmap_init_i2c(page, &pm886_regulator_regmap_config);
  344. if (IS_ERR(regmap))
  345. return dev_err_probe(dev, PTR_ERR(regmap),
  346. "Failed to initialize regulators regmap\n");
  347. rcfg.regmap = regmap;
  348. rcfg.dev = dev->parent;
  349. for (int i = 0; i < ARRAY_SIZE(pm886_regulators); i++) {
  350. rdesc = &pm886_regulators[i];
  351. rdev = devm_regulator_register(dev, rdesc, &rcfg);
  352. if (IS_ERR(rdev))
  353. return dev_err_probe(dev, PTR_ERR(rdev),
  354. "Failed to register %s\n", rdesc->name);
  355. }
  356. return 0;
  357. }
  358. static const struct platform_device_id pm886_regulator_id_table[] = {
  359. { "88pm886-regulator", },
  360. { }
  361. };
  362. MODULE_DEVICE_TABLE(platform, pm886_regulator_id_table);
  363. static struct platform_driver pm886_regulator_driver = {
  364. .driver = {
  365. .name = "88pm886-regulator",
  366. },
  367. .probe = pm886_regulator_probe,
  368. .id_table = pm886_regulator_id_table,
  369. };
  370. module_platform_driver(pm886_regulator_driver);
  371. MODULE_DESCRIPTION("Marvell 88PM886 PMIC regulator driver");
  372. MODULE_AUTHOR("Karel Balej <balejk@matfyz.cz>");
  373. MODULE_LICENSE("GPL");