fg_max17042.c 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2012 Samsung Electronics
  4. * Lukasz Majewski <l.majewski@samsung.com>
  5. */
  6. #include <common.h>
  7. #include <power/pmic.h>
  8. #include <power/max17042_fg.h>
  9. #include <i2c.h>
  10. #include <power/max8997_pmic.h>
  11. #include <power/power_chrg.h>
  12. #include <power/battery.h>
  13. #include <power/fg_battery_cell_params.h>
  14. #include <errno.h>
  15. static int fg_write_regs(struct pmic *p, u8 addr, u16 *data, int num)
  16. {
  17. int ret = 0;
  18. int i;
  19. for (i = 0; i < num; i++, addr++) {
  20. ret = pmic_reg_write(p, addr, *(data + i));
  21. if (ret)
  22. return ret;
  23. }
  24. return 0;
  25. }
  26. static int fg_read_regs(struct pmic *p, u8 addr, u16 *data, int num)
  27. {
  28. unsigned int dat;
  29. int ret = 0;
  30. int i;
  31. for (i = 0; i < num; i++, addr++) {
  32. ret = pmic_reg_read(p, addr, &dat);
  33. if (ret)
  34. return ret;
  35. *(data + i) = (u16)dat;
  36. }
  37. return 0;
  38. }
  39. static int fg_write_and_verify(struct pmic *p, u8 addr, u16 data)
  40. {
  41. unsigned int val = data;
  42. int ret = 0;
  43. ret |= pmic_reg_write(p, addr, val);
  44. ret |= pmic_reg_read(p, addr, &val);
  45. if (ret)
  46. return ret;
  47. if (((u16) val) == data)
  48. return 0;
  49. return -1;
  50. }
  51. static void por_fuelgauge_init(struct pmic *p)
  52. {
  53. u16 r_data0[16], r_data1[16], r_data2[16];
  54. u32 rewrite_count = 5;
  55. u32 check_count;
  56. u32 lock_count;
  57. u32 i = 0;
  58. u32 val;
  59. s32 ret = 0;
  60. char *status_msg;
  61. /* Delay 500 ms */
  62. mdelay(500);
  63. /* Initilize Configuration */
  64. pmic_reg_write(p, MAX17042_CONFIG, 0x2310);
  65. rewrite_model:
  66. check_count = 5;
  67. lock_count = 5;
  68. if (!rewrite_count--) {
  69. status_msg = "init failed!";
  70. goto error;
  71. }
  72. /* Unlock Model Access */
  73. pmic_reg_write(p, MAX17042_MLOCKReg1, MODEL_UNLOCK1);
  74. pmic_reg_write(p, MAX17042_MLOCKReg2, MODEL_UNLOCK2);
  75. /* Write/Read/Verify the Custom Model */
  76. ret = fg_write_regs(p, MAX17042_MODEL1, cell_character0,
  77. ARRAY_SIZE(cell_character0));
  78. if (ret)
  79. goto rewrite_model;
  80. ret = fg_write_regs(p, MAX17042_MODEL2, cell_character1,
  81. ARRAY_SIZE(cell_character1));
  82. if (ret)
  83. goto rewrite_model;
  84. ret = fg_write_regs(p, MAX17042_MODEL3, cell_character2,
  85. ARRAY_SIZE(cell_character2));
  86. if (ret)
  87. goto rewrite_model;
  88. check_model:
  89. if (!check_count--) {
  90. if (rewrite_count)
  91. goto rewrite_model;
  92. else
  93. status_msg = "check failed!";
  94. goto error;
  95. }
  96. ret = fg_read_regs(p, MAX17042_MODEL1, r_data0, ARRAY_SIZE(r_data0));
  97. if (ret)
  98. goto check_model;
  99. ret = fg_read_regs(p, MAX17042_MODEL2, r_data1, ARRAY_SIZE(r_data1));
  100. if (ret)
  101. goto check_model;
  102. ret = fg_read_regs(p, MAX17042_MODEL3, r_data2, ARRAY_SIZE(r_data2));
  103. if (ret)
  104. goto check_model;
  105. for (i = 0; i < 16; i++) {
  106. if ((cell_character0[i] != r_data0[i])
  107. || (cell_character1[i] != r_data1[i])
  108. || (cell_character2[i] != r_data2[i]))
  109. goto rewrite_model;
  110. }
  111. lock_model:
  112. if (!lock_count--) {
  113. if (rewrite_count)
  114. goto rewrite_model;
  115. else
  116. status_msg = "lock failed!";
  117. goto error;
  118. }
  119. /* Lock model access */
  120. pmic_reg_write(p, MAX17042_MLOCKReg1, MODEL_LOCK1);
  121. pmic_reg_write(p, MAX17042_MLOCKReg2, MODEL_LOCK2);
  122. /* Verify the model access is locked */
  123. ret = fg_read_regs(p, MAX17042_MODEL1, r_data0, ARRAY_SIZE(r_data0));
  124. if (ret)
  125. goto lock_model;
  126. ret = fg_read_regs(p, MAX17042_MODEL2, r_data1, ARRAY_SIZE(r_data1));
  127. if (ret)
  128. goto lock_model;
  129. ret = fg_read_regs(p, MAX17042_MODEL3, r_data2, ARRAY_SIZE(r_data2));
  130. if (ret)
  131. goto lock_model;
  132. for (i = 0; i < ARRAY_SIZE(r_data0); i++) {
  133. /* Check if model locked */
  134. if (r_data0[i] || r_data1[i] || r_data2[i])
  135. goto lock_model;
  136. }
  137. /* Write Custom Parameters */
  138. fg_write_and_verify(p, MAX17042_RCOMP0, RCOMP0);
  139. fg_write_and_verify(p, MAX17042_TEMPCO, TempCo);
  140. /* Delay at least 350mS */
  141. mdelay(350);
  142. /* Initialization Complete */
  143. pmic_reg_read(p, MAX17042_STATUS, &val);
  144. /* Write and Verify Status with POR bit Cleared */
  145. fg_write_and_verify(p, MAX17042_STATUS, val & ~MAX17042_POR);
  146. /* Delay at least 350 ms */
  147. mdelay(350);
  148. status_msg = "OK!";
  149. error:
  150. debug("%s: model init status: %s\n", p->name, status_msg);
  151. return;
  152. }
  153. static int power_update_battery(struct pmic *p, struct pmic *bat)
  154. {
  155. struct power_battery *pb = bat->pbat;
  156. unsigned int val;
  157. int ret = 0;
  158. if (pmic_probe(p)) {
  159. puts("Can't find max17042 fuel gauge\n");
  160. return -ENODEV;
  161. }
  162. ret |= pmic_reg_read(p, MAX17042_VFSOC, &val);
  163. pb->bat->state_of_chrg = (val >> 8);
  164. pmic_reg_read(p, MAX17042_VCELL, &val);
  165. debug("vfsoc: 0x%x\n", val);
  166. pb->bat->voltage_uV = ((val & 0xFFUL) >> 3) + ((val & 0xFF00) >> 3);
  167. pb->bat->voltage_uV = (pb->bat->voltage_uV * 625);
  168. pmic_reg_read(p, 0x05, &val);
  169. pb->bat->capacity = val >> 2;
  170. return ret;
  171. }
  172. static int power_check_battery(struct pmic *p, struct pmic *bat)
  173. {
  174. struct power_battery *pb = bat->pbat;
  175. unsigned int val;
  176. int ret = 0;
  177. if (pmic_probe(p)) {
  178. puts("Can't find max17042 fuel gauge\n");
  179. return -ENODEV;
  180. }
  181. ret |= pmic_reg_read(p, MAX17042_STATUS, &val);
  182. debug("fg status: 0x%x\n", val);
  183. if (val & MAX17042_POR)
  184. por_fuelgauge_init(p);
  185. ret |= pmic_reg_read(p, MAX17042_VERSION, &val);
  186. pb->bat->version = val;
  187. power_update_battery(p, bat);
  188. debug("fg ver: 0x%x\n", pb->bat->version);
  189. printf("BAT: state_of_charge(SOC):%d%%\n",
  190. pb->bat->state_of_chrg);
  191. printf(" voltage: %d.%6.6d [V] (expected to be %d [mAh])\n",
  192. pb->bat->voltage_uV / 1000000,
  193. pb->bat->voltage_uV % 1000000,
  194. pb->bat->capacity);
  195. if (pb->bat->voltage_uV > 3850000)
  196. pb->bat->state = EXT_SOURCE;
  197. else if (pb->bat->voltage_uV < 3600000 || pb->bat->state_of_chrg < 5)
  198. pb->bat->state = CHARGE;
  199. else
  200. pb->bat->state = NORMAL;
  201. return ret;
  202. }
  203. static struct power_fg power_fg_ops = {
  204. .fg_battery_check = power_check_battery,
  205. .fg_battery_update = power_update_battery,
  206. };
  207. int power_fg_init(unsigned char bus)
  208. {
  209. static const char name[] = "MAX17042_FG";
  210. struct pmic *p = pmic_alloc();
  211. if (!p) {
  212. printf("%s: POWER allocation error!\n", __func__);
  213. return -ENOMEM;
  214. }
  215. debug("Board Fuel Gauge init\n");
  216. p->name = name;
  217. p->interface = PMIC_I2C;
  218. p->number_of_regs = FG_NUM_OF_REGS;
  219. p->hw.i2c.addr = MAX17042_I2C_ADDR;
  220. p->hw.i2c.tx_num = 2;
  221. p->sensor_byte_order = PMIC_SENSOR_BYTE_ORDER_BIG;
  222. p->bus = bus;
  223. p->fg = &power_fg_ops;
  224. return 0;
  225. }