bq27220.c 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. #include "bq27220.h"
  2. #include "bq27220_reg.h"
  3. #include <furi-hal-delay.h>
  4. #include <furi/log.h>
  5. #include <stdbool.h>
  6. #define TAG "Gauge"
  7. uint16_t bq27220_read_word(FuriHalI2cBusHandle* handle, uint8_t address) {
  8. uint8_t buffer[2] = {address};
  9. uint16_t ret = 0;
  10. if(furi_hal_i2c_trx(handle, BQ27220_ADDRESS, buffer, 1, buffer, 2, BQ27220_I2C_TIMEOUT)) {
  11. ret = *(uint16_t*)buffer;
  12. }
  13. return ret;
  14. }
  15. bool bq27220_control(FuriHalI2cBusHandle* handle, uint16_t control) {
  16. bool ret = false;
  17. uint8_t buffer[3];
  18. buffer[0] = CommandControl;
  19. buffer[1] = control & 0xFF;
  20. buffer[2] = (control >> 8) & 0xFF;
  21. ret = furi_hal_i2c_tx(handle, BQ27220_ADDRESS, buffer, 3, BQ27220_I2C_TIMEOUT);
  22. return ret;
  23. }
  24. uint8_t bq27220_get_checksum(uint8_t* data, uint16_t len) {
  25. uint8_t ret = 0;
  26. for(uint16_t i = 0; i < len; i++) {
  27. ret += data[i];
  28. }
  29. return 0xFF - ret;
  30. }
  31. bool bq27220_set_parameter_u16(FuriHalI2cBusHandle* handle, uint16_t address, uint16_t value) {
  32. bool ret;
  33. uint8_t buffer[5];
  34. buffer[0] = CommandSelectSubclass;
  35. buffer[1] = address & 0xFF;
  36. buffer[2] = (address >> 8) & 0xFF;
  37. buffer[3] = (value >> 8) & 0xFF;
  38. buffer[4] = value & 0xFF;
  39. ret = furi_hal_i2c_tx(handle, BQ27220_ADDRESS, buffer, 5, BQ27220_I2C_TIMEOUT);
  40. delay_us(10000);
  41. uint8_t checksum = bq27220_get_checksum(&buffer[1], 4);
  42. buffer[0] = CommandMACDataSum;
  43. buffer[1] = checksum;
  44. buffer[2] = 6;
  45. ret = furi_hal_i2c_tx(handle, BQ27220_ADDRESS, buffer, 3, BQ27220_I2C_TIMEOUT);
  46. delay_us(10000);
  47. return ret;
  48. }
  49. bool bq27220_init(FuriHalI2cBusHandle* handle, const ParamCEDV* cedv) {
  50. uint32_t timeout = 100;
  51. uint16_t design_cap = bq27220_get_design_capacity(handle);
  52. if(cedv->design_cap == design_cap) {
  53. FURI_LOG_I(TAG, "Skip battery profile update");
  54. return true;
  55. }
  56. FURI_LOG_I(TAG, "Start updating battery profile");
  57. OperationStatus status = {};
  58. if(!bq27220_control(handle, Control_ENTER_CFG_UPDATE)) {
  59. FURI_LOG_E(TAG, "Can't configure update");
  60. return false;
  61. };
  62. while((status.CFGUPDATE != 1) && (timeout-- > 0)) {
  63. bq27220_get_operation_status(handle, &status);
  64. }
  65. bq27220_set_parameter_u16(handle, AddressGaugingConfig, cedv->cedv_conf.gauge_conf_raw);
  66. bq27220_set_parameter_u16(handle, AddressFullChargeCapacity, cedv->full_charge_cap);
  67. bq27220_set_parameter_u16(handle, AddressDesignCapacity, cedv->design_cap);
  68. bq27220_set_parameter_u16(handle, AddressEMF, cedv->EMF);
  69. bq27220_set_parameter_u16(handle, AddressC0, cedv->C0);
  70. bq27220_set_parameter_u16(handle, AddressR0, cedv->R0);
  71. bq27220_set_parameter_u16(handle, AddressT0, cedv->T0);
  72. bq27220_set_parameter_u16(handle, AddressR1, cedv->R1);
  73. bq27220_set_parameter_u16(handle, AddressTC, (cedv->TC) << 8 | cedv->C1);
  74. bq27220_set_parameter_u16(handle, AddressStartDOD0, cedv->DOD0);
  75. bq27220_set_parameter_u16(handle, AddressStartDOD10, cedv->DOD10);
  76. bq27220_set_parameter_u16(handle, AddressStartDOD20, cedv->DOD20);
  77. bq27220_set_parameter_u16(handle, AddressStartDOD30, cedv->DOD30);
  78. bq27220_set_parameter_u16(handle, AddressStartDOD40, cedv->DOD40);
  79. bq27220_set_parameter_u16(handle, AddressStartDOD50, cedv->DOD40);
  80. bq27220_set_parameter_u16(handle, AddressStartDOD60, cedv->DOD60);
  81. bq27220_set_parameter_u16(handle, AddressStartDOD70, cedv->DOD70);
  82. bq27220_set_parameter_u16(handle, AddressStartDOD80, cedv->DOD80);
  83. bq27220_set_parameter_u16(handle, AddressStartDOD90, cedv->DOD90);
  84. bq27220_set_parameter_u16(handle, AddressStartDOD100, cedv->DOD100);
  85. bq27220_set_parameter_u16(handle, AddressEDV0, cedv->EDV0);
  86. bq27220_set_parameter_u16(handle, AddressEDV1, cedv->EDV1);
  87. bq27220_set_parameter_u16(handle, AddressEDV2, cedv->EDV2);
  88. bq27220_control(handle, Control_EXIT_CFG_UPDATE);
  89. delay_us(10000);
  90. design_cap = bq27220_get_design_capacity(handle);
  91. if(cedv->design_cap == design_cap) {
  92. FURI_LOG_I(TAG, "Battery profile update success");
  93. return true;
  94. } else {
  95. FURI_LOG_E(TAG, "Battery profile update failed");
  96. return false;
  97. }
  98. }
  99. uint16_t bq27220_get_voltage(FuriHalI2cBusHandle* handle) {
  100. return bq27220_read_word(handle, CommandVoltage);
  101. }
  102. int16_t bq27220_get_current(FuriHalI2cBusHandle* handle) {
  103. return bq27220_read_word(handle, CommandCurrent);
  104. }
  105. uint8_t bq27220_get_battery_status(FuriHalI2cBusHandle* handle, BatteryStatus* battery_status) {
  106. uint16_t data = bq27220_read_word(handle, CommandBatteryStatus);
  107. if(data == BQ27220_ERROR) {
  108. return BQ27220_ERROR;
  109. } else {
  110. *(uint16_t*)battery_status = data;
  111. return BQ27220_SUCCESS;
  112. }
  113. }
  114. uint8_t bq27220_get_operation_status(FuriHalI2cBusHandle* handle, OperationStatus* operation_status) {
  115. uint16_t data = bq27220_read_word(handle, CommandOperationStatus);
  116. if(data == BQ27220_ERROR) {
  117. return BQ27220_ERROR;
  118. } else {
  119. *(uint16_t*)operation_status = data;
  120. return BQ27220_SUCCESS;
  121. }
  122. }
  123. uint16_t bq27220_get_temperature(FuriHalI2cBusHandle* handle) {
  124. return bq27220_read_word(handle, CommandTemperature);
  125. }
  126. uint16_t bq27220_get_full_charge_capacity(FuriHalI2cBusHandle* handle) {
  127. return bq27220_read_word(handle, CommandFullChargeCapacity);
  128. }
  129. uint16_t bq27220_get_design_capacity(FuriHalI2cBusHandle* handle) {
  130. return bq27220_read_word(handle, CommandDesignCapacity);
  131. }
  132. uint16_t bq27220_get_remaining_capacity(FuriHalI2cBusHandle* handle) {
  133. return bq27220_read_word(handle, CommandRemainingCapacity);
  134. }
  135. uint16_t bq27220_get_state_of_charge(FuriHalI2cBusHandle* handle) {
  136. return bq27220_read_word(handle, CommandStateOfCharge);
  137. }
  138. uint16_t bq27220_get_state_of_health(FuriHalI2cBusHandle* handle) {
  139. return bq27220_read_word(handle, CommandStateOfHealth);
  140. }