rtq2208-regulator.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668
  1. // SPDX-License-Identifier: GPL-2.0+
  2. #include <linux/bitops.h>
  3. #include <linux/bitfield.h>
  4. #include <linux/util_macros.h>
  5. #include <linux/module.h>
  6. #include <linux/i2c.h>
  7. #include <linux/regmap.h>
  8. #include <linux/regulator/driver.h>
  9. #include <linux/regulator/machine.h>
  10. #include <linux/regulator/of_regulator.h>
  11. #include <linux/mod_devicetable.h>
  12. /* Register */
  13. #define RTQ2208_REG_GLOBAL_INT1 0x12
  14. #define RTQ2208_REG_FLT_RECORDBUCK_CB 0x18
  15. #define RTQ2208_REG_GLOBAL_INT1_MASK 0x1D
  16. #define RTQ2208_REG_FLT_MASKBUCK_CB 0x1F
  17. #define RTQ2208_REG_BUCK_C_CFG0 0x32
  18. #define RTQ2208_REG_BUCK_B_CFG0 0x42
  19. #define RTQ2208_REG_BUCK_A_CFG0 0x52
  20. #define RTQ2208_REG_BUCK_D_CFG0 0x62
  21. #define RTQ2208_REG_BUCK_G_CFG0 0x72
  22. #define RTQ2208_REG_BUCK_F_CFG0 0x82
  23. #define RTQ2208_REG_BUCK_E_CFG0 0x92
  24. #define RTQ2208_REG_BUCK_H_CFG0 0xA2
  25. #define RTQ2208_REG_LDO1_CFG 0xB1
  26. #define RTQ2208_REG_LDO2_CFG 0xC1
  27. #define RTQ2208_REG_LDO_DVS_CTRL 0xD0
  28. #define RTQ2208_REG_HIDDEN_BUCKPH 0x55
  29. #define RTQ2208_REG_HIDDEN_LDOCFG0 0x8F
  30. #define RTQ2208_REG_HIDDEN_LDOCFG1 0x96
  31. #define RTQ2208_REG_HIDDEN0 0xFE
  32. #define RTQ2208_REG_HIDDEN1 0xFF
  33. /* Mask */
  34. #define RTQ2208_BUCK_NR_MTP_SEL_MASK GENMASK(7, 0)
  35. #define RTQ2208_BUCK_EN_NR_MTP_SEL0_MASK BIT(0)
  36. #define RTQ2208_BUCK_EN_NR_MTP_SEL1_MASK BIT(1)
  37. #define RTQ2208_BUCK_RSPUP_MASK GENMASK(6, 4)
  38. #define RTQ2208_BUCK_RSPDN_MASK GENMASK(2, 0)
  39. #define RTQ2208_BUCK_NRMODE_MASK BIT(5)
  40. #define RTQ2208_BUCK_STRMODE_MASK BIT(5)
  41. #define RTQ2208_BUCK_EN_STR_MASK BIT(0)
  42. #define RTQ2208_LDO_EN_STR_MASK BIT(7)
  43. #define RTQ2208_EN_DIS_MASK BIT(0)
  44. #define RTQ2208_BUCK_RAMP_SEL_MASK GENMASK(2, 0)
  45. #define RTQ2208_HD_INT_MASK BIT(0)
  46. #define RTQ2208_LDO1_DISCHG_EN_MASK BIT(4)
  47. #define RTQ2208_LDO1_VOSEL_SD_MASK BIT(5)
  48. #define RTQ2208_LDO2_DISCHG_EN_MASK BIT(6)
  49. #define RTQ2208_LDO2_VOSEL_SD_MASK BIT(7)
  50. #define RTQ2208_MASK_BUCKPH_GROUP1 GENMASK(6, 4)
  51. #define RTQ2208_MASK_BUCKPH_GROUP2 GENMASK(2, 0)
  52. #define RTQ2208_MASK_LDO2_OPT0 BIT(7)
  53. #define RTQ2208_MASK_LDO2_OPT1 BIT(7)
  54. #define RTQ2208_MASK_LDO1_FIXED BIT(6)
  55. /* Size */
  56. #define RTQ2208_VOUT_MAXNUM 256
  57. #define RTQ2208_BUCK_NUM_IRQ_REGS 5
  58. #define RTQ2208_STS_NUM_IRQ_REGS 2
  59. /* Value */
  60. #define RTQ2208_RAMP_VALUE_MIN_uV 500
  61. #define RTQ2208_RAMP_VALUE_MAX_uV 16000
  62. #define RTQ2208_BUCK_MASK(uv_irq, ov_irq) (1 << ((uv_irq) % 8) | 1 << ((ov_irq) % 8))
  63. enum {
  64. RTQ2208_BUCK_B = 0,
  65. RTQ2208_BUCK_C,
  66. RTQ2208_BUCK_D,
  67. RTQ2208_BUCK_A,
  68. RTQ2208_BUCK_F,
  69. RTQ2208_BUCK_G,
  70. RTQ2208_BUCK_H,
  71. RTQ2208_BUCK_E,
  72. RTQ2208_LDO2,
  73. RTQ2208_LDO1,
  74. RTQ2208_LDO_MAX,
  75. };
  76. enum {
  77. RTQ2208_AUTO_MODE = 0,
  78. RTQ2208_FCCM,
  79. };
  80. struct rtq2208_regulator_desc {
  81. struct regulator_desc desc;
  82. unsigned int mtp_sel_reg;
  83. unsigned int mtp_sel_mask;
  84. unsigned int mode_reg;
  85. unsigned int mode_mask;
  86. unsigned int suspend_config_reg;
  87. unsigned int suspend_enable_mask;
  88. unsigned int suspend_mode_mask;
  89. };
  90. struct rtq2208_rdev_map {
  91. struct regulator_dev *rdev[RTQ2208_LDO_MAX];
  92. struct regmap *regmap;
  93. struct device *dev;
  94. };
  95. /* set Normal Auto/FCCM mode */
  96. static int rtq2208_set_mode(struct regulator_dev *rdev, unsigned int mode)
  97. {
  98. const struct rtq2208_regulator_desc *rdesc =
  99. (const struct rtq2208_regulator_desc *)rdev->desc;
  100. unsigned int val, shift;
  101. switch (mode) {
  102. case REGULATOR_MODE_NORMAL:
  103. val = RTQ2208_AUTO_MODE;
  104. break;
  105. case REGULATOR_MODE_FAST:
  106. val = RTQ2208_FCCM;
  107. break;
  108. default:
  109. return -EINVAL;
  110. }
  111. shift = ffs(rdesc->mode_mask) - 1;
  112. return regmap_update_bits(rdev->regmap, rdesc->mode_reg,
  113. rdesc->mode_mask, val << shift);
  114. }
  115. static unsigned int rtq2208_get_mode(struct regulator_dev *rdev)
  116. {
  117. const struct rtq2208_regulator_desc *rdesc =
  118. (const struct rtq2208_regulator_desc *)rdev->desc;
  119. unsigned int mode_val;
  120. int ret;
  121. ret = regmap_read(rdev->regmap, rdesc->mode_reg, &mode_val);
  122. if (ret)
  123. return REGULATOR_MODE_INVALID;
  124. return (mode_val & rdesc->mode_mask) ? REGULATOR_MODE_FAST : REGULATOR_MODE_NORMAL;
  125. }
  126. static int rtq2208_set_ramp_delay(struct regulator_dev *rdev, int ramp_delay)
  127. {
  128. const struct regulator_desc *desc = rdev->desc;
  129. unsigned int sel = 0, val;
  130. ramp_delay = max(ramp_delay, RTQ2208_RAMP_VALUE_MIN_uV);
  131. ramp_delay = min(ramp_delay, RTQ2208_RAMP_VALUE_MAX_uV);
  132. ramp_delay /= RTQ2208_RAMP_VALUE_MIN_uV;
  133. /*
  134. * fls(ramp_delay) - 1: doing LSB shift, let it starts from 0
  135. *
  136. * RTQ2208_BUCK_RAMP_SEL_MASK - sel: doing descending order shifting.
  137. * Because the relation of seleltion and value is like that
  138. *
  139. * seletion: value
  140. * 010: 16mv
  141. * ...
  142. * 111: 0.5mv
  143. *
  144. * For example, if I would like to select 16mv, the fls(ramp_delay) - 1 will be 0b010,
  145. * and I need to use 0b111 - sel to do the shifting
  146. */
  147. sel = fls(ramp_delay) - 1;
  148. sel = RTQ2208_BUCK_RAMP_SEL_MASK - sel;
  149. val = FIELD_PREP(RTQ2208_BUCK_RSPUP_MASK, sel) | FIELD_PREP(RTQ2208_BUCK_RSPDN_MASK, sel);
  150. return regmap_update_bits(rdev->regmap, desc->ramp_reg,
  151. RTQ2208_BUCK_RSPUP_MASK | RTQ2208_BUCK_RSPDN_MASK, val);
  152. }
  153. static int rtq2208_set_suspend_enable(struct regulator_dev *rdev)
  154. {
  155. const struct rtq2208_regulator_desc *rdesc =
  156. (const struct rtq2208_regulator_desc *)rdev->desc;
  157. return regmap_set_bits(rdev->regmap, rdesc->suspend_config_reg, rdesc->suspend_enable_mask);
  158. }
  159. static int rtq2208_set_suspend_disable(struct regulator_dev *rdev)
  160. {
  161. const struct rtq2208_regulator_desc *rdesc =
  162. (const struct rtq2208_regulator_desc *)rdev->desc;
  163. return regmap_update_bits(rdev->regmap, rdesc->suspend_config_reg, rdesc->suspend_enable_mask, 0);
  164. }
  165. static int rtq2208_set_suspend_mode(struct regulator_dev *rdev, unsigned int mode)
  166. {
  167. const struct rtq2208_regulator_desc *rdesc =
  168. (const struct rtq2208_regulator_desc *)rdev->desc;
  169. unsigned int val, shift;
  170. switch (mode) {
  171. case REGULATOR_MODE_NORMAL:
  172. val = RTQ2208_AUTO_MODE;
  173. break;
  174. case REGULATOR_MODE_FAST:
  175. val = RTQ2208_FCCM;
  176. break;
  177. default:
  178. return -EINVAL;
  179. }
  180. shift = ffs(rdesc->suspend_mode_mask) - 1;
  181. return regmap_update_bits(rdev->regmap, rdesc->suspend_config_reg,
  182. rdesc->suspend_mode_mask, val << shift);
  183. }
  184. static const struct regulator_ops rtq2208_regulator_buck_ops = {
  185. .enable = regulator_enable_regmap,
  186. .disable = regulator_disable_regmap,
  187. .is_enabled = regulator_is_enabled_regmap,
  188. .list_voltage = regulator_list_voltage_linear_range,
  189. .set_voltage_sel = regulator_set_voltage_sel_regmap,
  190. .get_voltage_sel = regulator_get_voltage_sel_regmap,
  191. .set_mode = rtq2208_set_mode,
  192. .get_mode = rtq2208_get_mode,
  193. .set_ramp_delay = rtq2208_set_ramp_delay,
  194. .set_active_discharge = regulator_set_active_discharge_regmap,
  195. .set_suspend_enable = rtq2208_set_suspend_enable,
  196. .set_suspend_disable = rtq2208_set_suspend_disable,
  197. .set_suspend_mode = rtq2208_set_suspend_mode,
  198. };
  199. static const struct regulator_ops rtq2208_regulator_ldo_fix_ops = {
  200. .enable = regulator_enable_regmap,
  201. .disable = regulator_disable_regmap,
  202. .is_enabled = regulator_is_enabled_regmap,
  203. .set_active_discharge = regulator_set_active_discharge_regmap,
  204. .set_suspend_enable = rtq2208_set_suspend_enable,
  205. .set_suspend_disable = rtq2208_set_suspend_disable,
  206. };
  207. static const struct regulator_ops rtq2208_regulator_ldo_adj_ops = {
  208. .enable = regulator_enable_regmap,
  209. .disable = regulator_disable_regmap,
  210. .is_enabled = regulator_is_enabled_regmap,
  211. .list_voltage = regulator_list_voltage_table,
  212. .set_voltage_sel = regulator_set_voltage_sel_regmap,
  213. .get_voltage_sel = regulator_get_voltage_sel_regmap,
  214. .set_active_discharge = regulator_set_active_discharge_regmap,
  215. .set_suspend_enable = rtq2208_set_suspend_enable,
  216. .set_suspend_disable = rtq2208_set_suspend_disable,
  217. };
  218. static const unsigned int rtq2208_ldo_volt_table[] = {
  219. 1800000,
  220. 3300000,
  221. };
  222. static unsigned int rtq2208_of_map_mode(unsigned int mode)
  223. {
  224. switch (mode) {
  225. case RTQ2208_AUTO_MODE:
  226. return REGULATOR_MODE_NORMAL;
  227. case RTQ2208_FCCM:
  228. return REGULATOR_MODE_FAST;
  229. default:
  230. return REGULATOR_MODE_INVALID;
  231. }
  232. }
  233. static int rtq2208_init_irq_mask(struct rtq2208_rdev_map *rdev_map, unsigned int *buck_masks)
  234. {
  235. unsigned char buck_clr_masks[5] = {0x33, 0x33, 0x33, 0x33, 0x33},
  236. sts_clr_masks[2] = {0xE7, 0xF7}, sts_masks[2] = {0xE6, 0xF6};
  237. int ret;
  238. /* write clear all buck irq once */
  239. ret = regmap_bulk_write(rdev_map->regmap, RTQ2208_REG_FLT_RECORDBUCK_CB, buck_clr_masks, 5);
  240. if (ret)
  241. return dev_err_probe(rdev_map->dev, ret, "Failed to clr buck irqs\n");
  242. /* write clear general irq once */
  243. ret = regmap_bulk_write(rdev_map->regmap, RTQ2208_REG_GLOBAL_INT1, sts_clr_masks, 2);
  244. if (ret)
  245. return dev_err_probe(rdev_map->dev, ret, "Failed to clr general irqs\n");
  246. /* unmask buck ov/uv irq */
  247. ret = regmap_bulk_write(rdev_map->regmap, RTQ2208_REG_FLT_MASKBUCK_CB, buck_masks, 5);
  248. if (ret)
  249. return dev_err_probe(rdev_map->dev, ret, "Failed to unmask buck irqs\n");
  250. /* unmask needed general irq */
  251. return regmap_bulk_write(rdev_map->regmap, RTQ2208_REG_GLOBAL_INT1_MASK, sts_masks, 2);
  252. }
  253. static irqreturn_t rtq2208_irq_handler(int irqno, void *devid)
  254. {
  255. unsigned char buck_flags[RTQ2208_BUCK_NUM_IRQ_REGS], sts_flags[RTQ2208_STS_NUM_IRQ_REGS];
  256. int ret = 0, i, uv_bit, ov_bit;
  257. struct rtq2208_rdev_map *rdev_map = devid;
  258. struct regulator_dev *rdev;
  259. if (!rdev_map)
  260. return IRQ_NONE;
  261. /* read irq event */
  262. ret = regmap_bulk_read(rdev_map->regmap, RTQ2208_REG_FLT_RECORDBUCK_CB,
  263. buck_flags, ARRAY_SIZE(buck_flags));
  264. if (ret)
  265. return IRQ_NONE;
  266. ret = regmap_bulk_read(rdev_map->regmap, RTQ2208_REG_GLOBAL_INT1,
  267. sts_flags, ARRAY_SIZE(sts_flags));
  268. if (ret)
  269. return IRQ_NONE;
  270. /* clear irq event */
  271. ret = regmap_bulk_write(rdev_map->regmap, RTQ2208_REG_FLT_RECORDBUCK_CB,
  272. buck_flags, ARRAY_SIZE(buck_flags));
  273. if (ret)
  274. return IRQ_NONE;
  275. ret = regmap_bulk_write(rdev_map->regmap, RTQ2208_REG_GLOBAL_INT1,
  276. sts_flags, ARRAY_SIZE(sts_flags));
  277. if (ret)
  278. return IRQ_NONE;
  279. for (i = 0; i < RTQ2208_LDO_MAX; i++) {
  280. if (!rdev_map->rdev[i])
  281. continue;
  282. rdev = rdev_map->rdev[i];
  283. /* uv irq */
  284. uv_bit = (i & 1) ? 4 : 0;
  285. if (buck_flags[i >> 1] & (1 << uv_bit))
  286. regulator_notifier_call_chain(rdev,
  287. REGULATOR_EVENT_UNDER_VOLTAGE, NULL);
  288. /* ov irq */
  289. ov_bit = uv_bit + 1;
  290. if (buck_flags[i >> 1] & (1 << ov_bit))
  291. regulator_notifier_call_chain(rdev,
  292. REGULATOR_EVENT_REGULATION_OUT, NULL);
  293. /* hd irq */
  294. if (sts_flags[1] & RTQ2208_HD_INT_MASK)
  295. regulator_notifier_call_chain(rdev,
  296. REGULATOR_EVENT_OVER_TEMP, NULL);
  297. }
  298. return IRQ_HANDLED;
  299. }
  300. #define BUCK_INFO(_name, _id) \
  301. { \
  302. .name = _name, \
  303. .base = RTQ2208_REG_BUCK_##_id##_CFG0, \
  304. .enable_reg = BUCK_RG_SHIFT(RTQ2208_REG_BUCK_##_id##_CFG0, 2), \
  305. .dis_reg = RTQ2208_REG_BUCK_##_id##_CFG0, \
  306. }
  307. #define LDO_INFO(_name, _id) \
  308. { \
  309. .name = _name, \
  310. .base = RTQ2208_REG_LDO##_id##_CFG, \
  311. .enable_reg = RTQ2208_REG_LDO##_id##_CFG, \
  312. .dis_mask = RTQ2208_LDO##_id##_DISCHG_EN_MASK, \
  313. .dis_on = RTQ2208_LDO##_id##_DISCHG_EN_MASK, \
  314. .vsel_mask = RTQ2208_LDO##_id##_VOSEL_SD_MASK, \
  315. }
  316. #define BUCK_RG_SHIFT(_base, _shift) (_base + _shift)
  317. #define VSEL_SHIFT(_sel) (_sel ? 3 : 1)
  318. #define MTP_SEL_MASK(_sel) RTQ2208_BUCK_EN_NR_MTP_SEL##_sel##_MASK
  319. static const struct linear_range rtq2208_vout_range[] = {
  320. REGULATOR_LINEAR_RANGE(400000, 0, 180, 5000),
  321. REGULATOR_LINEAR_RANGE(1310000, 181, 255, 10000),
  322. };
  323. static void rtq2208_init_regulator_desc(struct rtq2208_regulator_desc *rdesc, int mtp_sel, int idx,
  324. unsigned int ldo1_fixed, unsigned int ldo2_fixed)
  325. {
  326. struct regulator_desc *desc;
  327. unsigned int fixed_uV;
  328. static const struct {
  329. char *name;
  330. int base;
  331. int enable_reg;
  332. int dis_reg;
  333. int dis_mask;
  334. int dis_on;
  335. int vsel_mask;
  336. } regulator_info[] = {
  337. BUCK_INFO("buck-b", B),
  338. BUCK_INFO("buck-c", C),
  339. BUCK_INFO("buck-d", D),
  340. BUCK_INFO("buck-a", A),
  341. BUCK_INFO("buck-f", F),
  342. BUCK_INFO("buck-g", G),
  343. BUCK_INFO("buck-h", H),
  344. BUCK_INFO("buck-e", E),
  345. LDO_INFO("ldo2", 2),
  346. LDO_INFO("ldo1", 1),
  347. }, *curr_info;
  348. curr_info = regulator_info + idx;
  349. desc = &rdesc->desc;
  350. desc->name = curr_info->name;
  351. desc->of_match = of_match_ptr(curr_info->name);
  352. desc->regulators_node = of_match_ptr("regulators");
  353. desc->id = idx;
  354. desc->owner = THIS_MODULE;
  355. desc->type = REGULATOR_VOLTAGE;
  356. desc->enable_mask = mtp_sel ? MTP_SEL_MASK(1) : MTP_SEL_MASK(0);
  357. desc->enable_reg = curr_info->enable_reg;
  358. desc->active_discharge_off = 0;
  359. rdesc->mode_mask = RTQ2208_BUCK_NRMODE_MASK;
  360. switch (idx) {
  361. case RTQ2208_BUCK_B ... RTQ2208_BUCK_E:
  362. /* init buck desc */
  363. desc->ops = &rtq2208_regulator_buck_ops;
  364. desc->vsel_reg = curr_info->base + VSEL_SHIFT(mtp_sel);
  365. desc->vsel_mask = RTQ2208_BUCK_NR_MTP_SEL_MASK;
  366. desc->n_voltages = RTQ2208_VOUT_MAXNUM;
  367. desc->linear_ranges = rtq2208_vout_range;
  368. desc->n_linear_ranges = ARRAY_SIZE(rtq2208_vout_range);
  369. desc->ramp_reg = BUCK_RG_SHIFT(curr_info->base, 5);
  370. desc->of_map_mode = rtq2208_of_map_mode;
  371. desc->active_discharge_reg = curr_info->dis_reg;
  372. desc->active_discharge_on = RTQ2208_EN_DIS_MASK;
  373. desc->active_discharge_mask = RTQ2208_EN_DIS_MASK;
  374. rdesc->mode_reg = BUCK_RG_SHIFT(curr_info->base, 2);
  375. rdesc->suspend_config_reg = BUCK_RG_SHIFT(curr_info->base, 4);
  376. rdesc->suspend_enable_mask = RTQ2208_BUCK_EN_STR_MASK;
  377. rdesc->suspend_mode_mask = RTQ2208_BUCK_STRMODE_MASK;
  378. break;
  379. default:
  380. fixed_uV = idx == RTQ2208_LDO2 ? ldo2_fixed : ldo1_fixed;
  381. if (fixed_uV) {
  382. desc->n_voltages = 1;
  383. desc->fixed_uV = fixed_uV;
  384. desc->ops = &rtq2208_regulator_ldo_fix_ops;
  385. } else {
  386. desc->n_voltages = ARRAY_SIZE(rtq2208_ldo_volt_table);
  387. desc->volt_table = rtq2208_ldo_volt_table;
  388. desc->ops = &rtq2208_regulator_ldo_adj_ops;
  389. }
  390. /* init ldo desc */
  391. desc->active_discharge_reg = RTQ2208_REG_LDO_DVS_CTRL;
  392. desc->active_discharge_on = curr_info->dis_on;
  393. desc->active_discharge_mask = curr_info->dis_mask;
  394. desc->vsel_reg = RTQ2208_REG_LDO_DVS_CTRL;
  395. desc->vsel_mask = curr_info->vsel_mask;
  396. rdesc->suspend_config_reg = curr_info->base;
  397. rdesc->suspend_enable_mask = RTQ2208_LDO_EN_STR_MASK;
  398. break;
  399. }
  400. }
  401. static int rtq2208_parse_regulator_dt_data(int n_regulator, const unsigned int *regulator_idx_table,
  402. struct rtq2208_regulator_desc *rdesc[RTQ2208_LDO_MAX], struct device *dev,
  403. unsigned int ldo1_fixed, unsigned int ldo2_fixed)
  404. {
  405. int mtp_sel, i, idx;
  406. /* get mtp_sel0 or mtp_sel1 */
  407. mtp_sel = device_property_read_bool(dev, "richtek,mtp-sel-high");
  408. for (i = 0; i < n_regulator; i++) {
  409. idx = regulator_idx_table[i];
  410. rdesc[i] = devm_kcalloc(dev, 1, sizeof(*rdesc[0]), GFP_KERNEL);
  411. if (!rdesc[i])
  412. return -ENOMEM;
  413. rtq2208_init_regulator_desc(rdesc[i], mtp_sel, idx, ldo1_fixed, ldo2_fixed);
  414. }
  415. return 0;
  416. }
  417. static int rtq2208_regulator_check(struct device *dev, int *num, int *regulator_idx_table,
  418. unsigned int *buck_masks, unsigned int *ldo1_fixed_uV,
  419. unsigned int *ldo2_fixed_uV)
  420. {
  421. struct regmap *regmap = dev_get_regmap(dev, NULL);
  422. bool rtq2208_used_table[RTQ2208_LDO_MAX] = {0};
  423. u8 entry_key[] = { 0x69, 0x01 };
  424. unsigned int buck_phase, ldo_cfg0, ldo_cfg1;
  425. int i, ret;
  426. u8 mask;
  427. ret = regmap_raw_write(regmap, RTQ2208_REG_HIDDEN0, entry_key, ARRAY_SIZE(entry_key));
  428. if (ret)
  429. return dev_err_probe(dev, ret, "Failed to enter hidden page\n");
  430. ret = regmap_read(regmap, RTQ2208_REG_HIDDEN_BUCKPH, &buck_phase);
  431. if (ret)
  432. return dev_err_probe(dev, ret, "Failed to read buck phase configuration\n");
  433. ret = regmap_read(regmap, RTQ2208_REG_HIDDEN_LDOCFG0, &ldo_cfg0);
  434. if (ret)
  435. return dev_err_probe(dev, ret, "Failed to read ldo cfg0\n");
  436. ret = regmap_read(regmap, RTQ2208_REG_HIDDEN_LDOCFG1, &ldo_cfg1);
  437. if (ret)
  438. return dev_err_probe(dev, ret, "Failed to read ldo cfg1\n");
  439. ret = regmap_write(regmap, RTQ2208_REG_HIDDEN1, 0x00);
  440. if (ret)
  441. return dev_err_probe(dev, ret, "Failed to exit hidden page\n");
  442. dev_info(dev, "BUCK Phase 0x%x\n", buck_phase);
  443. /*
  444. * Use buck phase configuration to assign used table mask
  445. * GROUP1 GROUP2
  446. * 0 -> 2P + 2P BC FG
  447. * 1 -> 2P + 1P + 1P BCA FGE
  448. * 2 -> 1P + 1P + 1P + 1P BCDA FGHE
  449. * 3 -> 3P + 1P BC FG
  450. * others -> 4P C G
  451. */
  452. switch (FIELD_GET(RTQ2208_MASK_BUCKPH_GROUP1, buck_phase)) {
  453. case 2:
  454. rtq2208_used_table[RTQ2208_BUCK_D] = true;
  455. fallthrough;
  456. case 1:
  457. rtq2208_used_table[RTQ2208_BUCK_A] = true;
  458. fallthrough;
  459. case 0:
  460. case 3:
  461. rtq2208_used_table[RTQ2208_BUCK_B] = true;
  462. fallthrough;
  463. default:
  464. rtq2208_used_table[RTQ2208_BUCK_C] = true;
  465. break;
  466. }
  467. switch (FIELD_GET(RTQ2208_MASK_BUCKPH_GROUP2, buck_phase)) {
  468. case 2:
  469. rtq2208_used_table[RTQ2208_BUCK_H] = true;
  470. fallthrough;
  471. case 1:
  472. rtq2208_used_table[RTQ2208_BUCK_E] = true;
  473. fallthrough;
  474. case 0:
  475. case 3:
  476. rtq2208_used_table[RTQ2208_BUCK_F] = true;
  477. fallthrough;
  478. default:
  479. rtq2208_used_table[RTQ2208_BUCK_G] = true;
  480. break;
  481. }
  482. *ldo1_fixed_uV = FIELD_GET(RTQ2208_MASK_LDO1_FIXED, ldo_cfg1) ? 1200000 : 0;
  483. if (!FIELD_GET(RTQ2208_MASK_LDO2_OPT0, ldo_cfg0) &&
  484. !FIELD_GET(RTQ2208_MASK_LDO2_OPT1, ldo_cfg1))
  485. *ldo2_fixed_uV = 0;
  486. else if (FIELD_GET(RTQ2208_MASK_LDO2_OPT1, ldo_cfg1))
  487. *ldo2_fixed_uV = 900000;
  488. else
  489. *ldo2_fixed_uV = 1200000;
  490. /* By default, LDO1 & LDO2 are always used */
  491. rtq2208_used_table[RTQ2208_LDO1] = rtq2208_used_table[RTQ2208_LDO2] = true;
  492. for (i = 0; i < RTQ2208_LDO_MAX; i++) {
  493. if (!rtq2208_used_table[i])
  494. continue;
  495. regulator_idx_table[(*num)++] = i;
  496. mask = RTQ2208_BUCK_MASK(4 * i, 4 * i + 1);
  497. buck_masks[i >> 1] &= ~mask;
  498. }
  499. return 0;
  500. }
  501. static const struct regmap_config rtq2208_regmap_config = {
  502. .reg_bits = 8,
  503. .val_bits = 8,
  504. .max_register = 0xFF,
  505. };
  506. static int rtq2208_probe(struct i2c_client *i2c)
  507. {
  508. struct device *dev = &i2c->dev;
  509. struct regmap *regmap;
  510. struct rtq2208_regulator_desc *rdesc[RTQ2208_LDO_MAX];
  511. struct regulator_dev *rdev;
  512. struct regulator_config cfg = {};
  513. struct rtq2208_rdev_map *rdev_map;
  514. int i, ret = 0, idx, n_regulator = 0;
  515. unsigned int regulator_idx_table[RTQ2208_LDO_MAX],
  516. buck_masks[RTQ2208_BUCK_NUM_IRQ_REGS] = {0x33, 0x33, 0x33, 0x33, 0x33};
  517. unsigned int ldo1_fixed_uV, ldo2_fixed_uV;
  518. rdev_map = devm_kzalloc(dev, sizeof(struct rtq2208_rdev_map), GFP_KERNEL);
  519. if (!rdev_map)
  520. return -ENOMEM;
  521. regmap = devm_regmap_init_i2c(i2c, &rtq2208_regmap_config);
  522. if (IS_ERR(regmap))
  523. return dev_err_probe(dev, PTR_ERR(regmap), "Failed to allocate regmap\n");
  524. /* get needed regulator */
  525. ret = rtq2208_regulator_check(dev, &n_regulator, regulator_idx_table, buck_masks,
  526. &ldo1_fixed_uV, &ldo2_fixed_uV);
  527. if (ret)
  528. return dev_err_probe(dev, ret, "Failed to check used regulators\n");
  529. rdev_map->regmap = regmap;
  530. rdev_map->dev = dev;
  531. cfg.dev = dev;
  532. /* init regulator desc */
  533. ret = rtq2208_parse_regulator_dt_data(n_regulator, regulator_idx_table, rdesc, dev,
  534. ldo1_fixed_uV, ldo2_fixed_uV);
  535. if (ret)
  536. return ret;
  537. for (i = 0; i < n_regulator; i++) {
  538. idx = regulator_idx_table[i];
  539. /* register regulator */
  540. rdev = devm_regulator_register(dev, &rdesc[i]->desc, &cfg);
  541. if (IS_ERR(rdev))
  542. return PTR_ERR(rdev);
  543. rdev_map->rdev[idx] = rdev;
  544. }
  545. /* init interrupt mask */
  546. ret = rtq2208_init_irq_mask(rdev_map, buck_masks);
  547. if (ret)
  548. return ret;
  549. /* register interrupt */
  550. return devm_request_threaded_irq(dev, i2c->irq, NULL, rtq2208_irq_handler,
  551. IRQF_ONESHOT, dev_name(dev), rdev_map);
  552. }
  553. static const struct of_device_id rtq2208_device_tables[] = {
  554. { .compatible = "richtek,rtq2208" },
  555. {}
  556. };
  557. MODULE_DEVICE_TABLE(of, rtq2208_device_tables);
  558. static struct i2c_driver rtq2208_driver = {
  559. .driver = {
  560. .name = "rtq2208",
  561. .of_match_table = rtq2208_device_tables,
  562. },
  563. .probe = rtq2208_probe,
  564. };
  565. module_i2c_driver(rtq2208_driver);
  566. MODULE_AUTHOR("Alina Yu <alina_yu@richtek.com>");
  567. MODULE_DESCRIPTION("Richtek RTQ2208 Regulator Driver");
  568. MODULE_LICENSE("GPL");