/* MIT License Copyright (c) 2019-2021 Mika Tuupola Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. -cut- This file is part of hardware agnostic I2C driver for AXP192: https://github.com/tuupola/axp192 SPDX-License-Identifier: MIT Version: 0.6.0 */ #include #include #include "axp192_config.h" #include "axp192.h" static axp192_err_t read_coloumb_counter(const axp192_t *axp, float *buffer); static axp192_err_t read_battery_power(const axp192_t *axp, float *buffer); static const axp192_init_command_t init_commands[] = { #ifdef AXP192_INCLUDE_SDKCONFIG_H /* Currently you have to use menuconfig to be able to use axp192_init() */ {AXP192_DCDC1_VOLTAGE, {CONFIG_AXP192_DCDC1_VOLTAGE}, 1}, {AXP192_DCDC3_VOLTAGE, {CONFIG_AXP192_DCDC3_VOLTAGE}, 1}, {AXP192_LDO23_VOLTAGE, {CONFIG_AXP192_LDO23_VOLTAGE}, 1}, {AXP192_GPIO0_LDOIO0_VOLTAGE, {CONFIG_AXP192_GPIO0_LDOIO0_VOLTAGE}, 1}, {AXP192_DCDC13_LDO23_CONTROL, {CONFIG_AXP192_DCDC13_LDO23_CONTROL}, 1}, {AXP192_EXTEN_DCDC2_CONTROL, {CONFIG_AXP192_EXTEN_DCDC2_CONTROL}, 1}, {AXP192_GPIO0_CONTROL, {CONFIG_AXP192_GPIO0_CONTROL}, 1}, {AXP192_GPIO1_CONTROL, {CONFIG_AXP192_GPIO1_CONTROL}, 1}, {AXP192_GPIO2_CONTROL, {CONFIG_AXP192_GPIO2_CONTROL}, 1}, {AXP192_GPIO43_FUNCTION_CONTROL, {CONFIG_AXP192_GPIO43_FUNCTION_CONTROL}, 1}, {AXP192_ADC_ENABLE_1, {CONFIG_AXP192_ADC_ENABLE_1}, 1}, {AXP192_CHARGE_CONTROL_1, {CONFIG_AXP192_CHARGE_CONTROL_1}, 1}, {AXP192_BATTERY_CHARGE_CONTROL, {CONFIG_AXP192_BATTERY_CHARGE_CONTROL}, 1}, #endif /* AXP192_INCLUDE_SDKCONFIG_H */ /* End of commands. */ {0, {0}, 0xff}, }; axp192_err_t axp192_init(const axp192_t *axp) { uint8_t cmd = 0; axp192_err_t status; /* Send all the commands. */ while (init_commands[cmd].count != 0xff) { status = axp->write( axp->handle, AXP192_ADDRESS, init_commands[cmd].command, init_commands[cmd].data, init_commands[cmd].count & 0x1f ); if (AXP192_OK != status) { return status; } cmd++; } return AXP192_OK; } static axp192_err_t axp192_read_adc(const axp192_t *axp, uint8_t reg, float *buffer) { uint8_t tmp[4]; float sensitivity = 1.0; float offset = 0.0; axp192_err_t status; switch (reg) { case AXP192_ACIN_VOLTAGE: case AXP192_VBUS_VOLTAGE: /* 1.7mV per LSB */ sensitivity = 1.7 / 1000; break; case AXP192_ACIN_CURRENT: /* 0.375mA per LSB */ sensitivity = 0.625 / 1000; break; case AXP192_VBUS_CURRENT: /* 0.375mA per LSB */ sensitivity = 0.375 / 1000; break; case AXP192_TEMP: /* 0.1C per LSB, 0x00 = -144.7C */ sensitivity = 0.1; offset = -144.7; break; case AXP192_TS_INPUT: /* 0.8mV per LSB */ sensitivity = 0.8 / 1000; break; case AXP192_BATTERY_POWER: /* 1.1mV * 0.5mA per LSB */ return read_battery_power(axp, buffer); break; case AXP192_BATTERY_VOLTAGE: /* 1.1mV per LSB */ sensitivity = 1.1 / 1000; break; case AXP192_CHARGE_CURRENT: case AXP192_DISCHARGE_CURRENT: /* 0.5mV per LSB */ sensitivity = 0.5 / 1000; break; case AXP192_APS_VOLTAGE: /* 1.4mV per LSB */ sensitivity = 1.4 / 1000; break; case AXP192_COULOMB_COUNTER: /* This is currently untested. */ return read_coloumb_counter(axp, buffer); break; } status = axp->read(axp->handle, AXP192_ADDRESS, reg, tmp, 2); if (AXP192_OK != status) { return status; } *buffer = (((tmp[0] << 4) + tmp[1]) * sensitivity) + offset; return AXP192_OK; } axp192_err_t axp192_read(const axp192_t *axp, uint8_t reg, void *buffer) { switch (reg) { case AXP192_ACIN_VOLTAGE: case AXP192_VBUS_VOLTAGE: case AXP192_ACIN_CURRENT: case AXP192_VBUS_CURRENT: case AXP192_TEMP: case AXP192_TS_INPUT: case AXP192_BATTERY_POWER: case AXP192_BATTERY_VOLTAGE: case AXP192_CHARGE_CURRENT: case AXP192_DISCHARGE_CURRENT: case AXP192_APS_VOLTAGE: case AXP192_COULOMB_COUNTER: /* Return ADC value. */ return axp192_read_adc(axp, reg, buffer); break; default: /* Return raw register value. */ return axp->read(axp->handle, AXP192_ADDRESS, reg, buffer, 1); } } axp192_err_t axp192_write(const axp192_t *axp, uint8_t reg, uint8_t value) { switch (reg) { case AXP192_ACIN_VOLTAGE: case AXP192_VBUS_VOLTAGE: case AXP192_ACIN_CURRENT: case AXP192_VBUS_CURRENT: case AXP192_TEMP: case AXP192_TS_INPUT: case AXP192_BATTERY_POWER: case AXP192_BATTERY_VOLTAGE: case AXP192_CHARGE_CURRENT: case AXP192_DISCHARGE_CURRENT: case AXP192_APS_VOLTAGE: case AXP192_COULOMB_COUNTER: /* Read only register. */ return AXP192_ERROR_ENOTSUP; break; default: /* Write raw register value. */ return axp->write(axp->handle, AXP192_ADDRESS, reg, &value, 1); } } axp192_err_t axp192_ioctl(const axp192_t *axp, int command, ...) { uint8_t reg = command >> 8; uint8_t tmp; uint16_t argument; va_list ap; switch (command) { case AXP192_COULOMB_COUNTER_ENABLE: tmp = 0b10000000; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_COULOMB_COUNTER_DISABLE: tmp = 0b00000000; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_COULOMB_COUNTER_SUSPEND: tmp = 0b11000000; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_COULOMB_COUNTER_CLEAR: tmp = 0b10100000; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_LDOIO0_ENABLE: /* 0x02 = LDO */ tmp = 0b00000010; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_LDOIO0_DISABLE: /* 0x07 = float */ tmp = 0b00000111; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_LDO2_ENABLE: /* This is currently untested. */ case AXP192_EXTEN_ENABLE: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp |= 0b00000100; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_LDO2_DISABLE: /* This is currently untested. */ case AXP192_EXTEN_DISABLE: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp &= ~0b00000100; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_GPIO2_SET_LEVEL: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); va_start(ap, command); argument = (uint8_t) va_arg(ap, int); va_end(ap); if (argument) { tmp |= 0b00000100; } else { tmp &= ~0b00000100; } return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_LDO3_ENABLE: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp |= 0b00001000; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_LDO3_DISABLE: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp &= ~0b00001000; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_DCDC1_ENABLE: /* This is currently untested. */ case AXP192_DCDC2_ENABLE: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp |= 0b00000001; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_DCDC1_DISABLE: /* This is currently untested. */ case AXP192_DCDC2_DISABLE: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp &= ~0b00000001; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_DCDC3_ENABLE: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp |= 0b00000010; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_DCDC3_DISABLE: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp &= ~0b00000010; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_GPIO1_SET_LEVEL: case AXP192_GPIO4_SET_LEVEL: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); va_start(ap, command); argument = (uint8_t) va_arg(ap, int); va_end(ap); if (argument) { tmp |= 0b00000010; } else { tmp &= ~0b00000010; } return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_GPIO0_SET_LEVEL: axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); va_start(ap, command); argument = (uint8_t) va_arg(ap, int); va_end(ap); if (argument) { tmp |= 0b00000001; } else { tmp &= ~0b00000001; } return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; case AXP192_DCDC1_SET_VOLTAGE: case AXP192_DCDC3_SET_VOLTAGE: va_start(ap, command); argument = (uint16_t) va_arg(ap, int); va_end(ap); /* 700-3500mv 25mV per step */ if ((argument < 700) || (argument > 3500)) { return AXP192_ERROR_EINVAL; } tmp = (argument - 700) / 25; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_DCDC2_SET_VOLTAGE: va_start(ap, command); argument = (uint16_t) va_arg(ap, int); va_end(ap); /* 700-2275mV 25mV per step */ if ((argument < 700) || (argument > 2275)) { return AXP192_ERROR_EINVAL; } tmp = (argument - 700) / 25; return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_LDO2_SET_VOLTAGE: va_start(ap, command); argument = (uint16_t) va_arg(ap, int); va_end(ap); /* 1800-3300mV 100mV per step */ if ((argument < 1800) || (argument > 3300)) { return AXP192_ERROR_EINVAL; } axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp &= ~0xf0; tmp |= (((argument - 1800) / 100) << 4); return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_LDO3_SET_VOLTAGE: va_start(ap, command); argument = (uint16_t) va_arg(ap, int); va_end(ap); /* 1800-3300mV 100mV per step */ if ((argument < 1800) || (argument > 3300)) { return AXP192_ERROR_EINVAL; } axp->read(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); tmp &= ~0x0f; tmp |= ((argument - 1800) / 100); return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; /* This is currently untested. */ case AXP192_LDOIO0_SET_VOLTAGE: va_start(ap, command); argument = (uint16_t) va_arg(ap, int); va_end(ap); /* 1800-3300mV 100mV per step, 2800mV default. */ if ((argument < 1800) || (argument > 3300)) { return AXP192_ERROR_EINVAL; } tmp = (((argument - 1800) / 100) << 4); return axp->write(axp->handle, AXP192_ADDRESS, reg, &tmp, 1); break; } return AXP192_ERROR_NOTTY; } static axp192_err_t read_coloumb_counter(const axp192_t *axp, float *buffer) { uint8_t tmp[4]; int32_t coin, coout; axp192_err_t status; status = axp->read(axp->handle, AXP192_ADDRESS, AXP192_CHARGE_COULOMB, tmp, sizeof(coin)); if (AXP192_OK != status) { return status; } coin = (tmp[0] << 24) + (tmp[1] << 16) + (tmp[2] << 8) + tmp[3]; status = axp->read(axp->handle, AXP192_ADDRESS, AXP192_DISCHARGE_COULOMB, tmp, sizeof(coout)); if (AXP192_OK != status) { return status; } coout = (tmp[0] << 24) + (tmp[1] << 16) + (tmp[2] << 8) + tmp[3]; /* CmAh = 65536 * 0.5mA *(coin - cout) / 3600 / ADC sample rate */ *buffer = 32768 * (coin - coout) / 3600 / 25; return AXP192_OK; } static axp192_err_t read_battery_power(const axp192_t *axp, float *buffer) { uint8_t tmp[4]; float sensitivity; axp192_err_t status; /* 1.1mV * 0.5mA per LSB */ sensitivity = 1.1 * 0.5 / 1000; status = axp->read(axp->handle, AXP192_ADDRESS, AXP192_BATTERY_POWER, tmp, 3); if (AXP192_OK != status) { return status; } *buffer = (((tmp[0] << 16) + (tmp[1] << 8) + tmp[2]) * sensitivity); return AXP192_OK; }