Tasmota/tasmota/berry/axp192.be

480 lines
13 KiB
Plaintext

#-------------------------------------------------------------
- I2C_Driver class to simplify development of I2C drivers
-
- I2C_Driver(name, addr [, i2c_index]) -> nil
- name: name of I2C device for logging
- addr: I2C address of device, will probe all I2C buses for it
- i2c_index: (optional) check is the device is not disabled
-------------------------------------------------------------#
class I2C_Driver : Driver
var wire #- if wire == nil then the module is not initialized -#
var addr
var name
def init(name, addr, i2c_index)
import string
var tasmota = self.get_tasmota()
#- check if the i2c index is disabled by Tasmota configuration -#
if i2c_index != nil && !tasmota.i2c_enabled(i2c_index) return end
self.addr = addr #- address for AXP192 -#
self.wire = tasmota.wire_scan(self.addr) #- get the right I2C bus -#
self.name = name #- display name for I2C device -#
if self.wire
print(string.format("I2C: %s detected on bus %d", self.name, self.wire.bus))
end
end
#- verify if the device is what is expected -#
#- this method needs to be overriden -#
def verify()
return true
end
# def log_write1(reg, val)
# import string
# var tasmota = self.get_tasmota()
# tasmota.log(string.format("I2C: (0x%02X) write %02X:%02X", self.addr, reg, val))
# end
def write1(reg, val)
# self.log_write1(reg, val)
return self.wire.write(self.addr, reg, val, 1)
end
# Set or clear a specific bit in a register
# write_bit(reg:int, bit:int, state:bool) -> nil
# reg: I2C register number (0..255)
# bit: bit of I2C register to change (0..7)
# state: boolean value to write to specified bit
def write_bit(reg, bit, state)
if bit < 0 || bit > 7 return end
var mark = 1 << bit
if state self.write1(reg, self.read8(reg) | mark)
else self.write1(reg, self.read8(reg) & (0xFF - mark))
end
end
# read 8 bits
def read8(reg)
return self.wire.read(self.addr, reg, 1)
end
# read 12 bits
def read12(reg)
var buf = self.wire.read_bytes(self.addr, reg, 2)
return (buf[0] << 4) + buf[1]
end
# read 13 bits
def read13(reg)
var buf = self.wire.read_bytes(self.addr, reg, 2)
return (buf[0] << 5) + buf[1]
end
# read 24 bits
def read24(reg)
var buf = self.wire.read_bytes(self.addr, reg, 3)
return (buf[0] << 16) + (buf[1] << 8) + buf[2]
end
# read 32 bits
def read32(reg)
var buf = self.wire.read_bytes(self.addr, reg, 4)
return (buf[0] << 24) + (buf[1] << 16) + (buf[2] << 8) + buf[3]
end
end
#-------------------------------------------------------------
- Generic driver for AXP192
-------------------------------------------------------------#
class AXP192 : I2C_Driver
def init()
super(self).init("AXP192", 0x34)
end
# Return True = Battery Exist
def battery_present()
if self.wire.read(self.addr, 0x01, 1) & 0x20 return true
else return false
end
end
# Input Power Status ???
def get_input_power_status()
return self.wire.read(self.addr, 0x00, 1)
end
# Battery Charging Status
def get_battery_chargin_status()
return self.wire.read(self.addr, 0x01, 1)
end
# AXP chip temperature in °C
def get_temp()
return self.read12(0x5E) * 0.1 - 144.7
end
def get_bat_power()
return self.read24(0x70) * 0.00055
end
def get_bat_voltage()
return self.read12(0x78) * 0.0011
end
def get_bat_current()
return (self.read13(0x7A) - self.read13(0x7C)) * 0.5
end
def get_bat_charge_current()
return self.read13(0x7A) * 0.5
end
def get_aps_voltage()
return self.read12(0x7E) * 0.0014
end
def get_vbus_voltage()
return self.read12(0x5A) * 0.0017
end
def get_vbus_current()
return self.read12(0x5C) * 0.375
end
# set LDO voltage
# ldo: 2/3
# voltage: (mV) 1800mV - 3300mV in 100mV steps
def set_ldo_voltage(ldo, voltage)
if voltage > 3300 voltage = 15
else voltage = (voltage / 100) - 18
end
if ldo == 2
self.write1(0x28, self.read8(0x28) & 0x0F | ((voltage & 0x0F) << 4))
end
if ldo == 3
self.write1(0x28, self.read8(0x28) & 0xF0 | (voltage & 0x0F))
end
end
# set DCDC enable, 1/2/3
def set_dcdc_enable(dcdc, state)
if dcdc == 1 self.write_bit(0x12, 0, state) end
if dcdc == 2 self.write_bit(0x12, 4, state) end
if dcdc == 3 self.write_bit(0x12, 1, state) end
end
# set LDO enable, 2/3 (LDO 1 is always on)
def set_ldo_enable(ldo, state)
if ldo == 2 self.write_bit(0x12, 2, state) end
if ldo == 3 self.write_bit(0x12, 3, state) end
end
# set GPIO output state 0/1/2 and 3/4
def write_gpio(gpio, state)
if gpio >= 0 && gpio <= 2
self.write_bit(0x94, gpio, state)
elif gpio >= 3 && gpio <= 4
self.write_bit(0x96, gpio - 3, state)
end
end
# Set voltage on DC-DC1/2/3
# dcdc: 1/2/3 (warning some C libs start at 0)
# voltage:
def set_dc_voltage(dcdc, voltage)
if dcdc < 1 || dcdc > 3 return end
var v
if voltage < 700 v = 0
elif voltage > 3500 v = 112
elif dcdc == 2 && voltage > 2275 v = 63 # dcdc2 is limited to 2.275V
else v = (voltage - 700) / 25
end
var addr = 0x26
if dcdc == 3 addr = 0x27
elif dcdc == 2 addr = 0x23
end
self.write1(addr, self.read8(addr) & 0x80 | (v & 0x7F))
end
# Set charging current
# 100mA = 0
# 190mA = 1
# 280mA = 2
# 360mA = 3
# 450mA = 4
# 550mA = 5
# 630mA = 6
# 700mA = 7
# 780mA = 8
# 880mA = 9
# 960mA = 10
# 1000mA = 11
# 1080mA = 12
# 1160mA = 13
# 1240mA = 14
# 1320mA = 15
def set_chg_current(current_code)
self.write1(0x33, self.read8(0x33) & 0xF0 | (current_code & 0x0F))
end
# // Low Volt Level 1, when APS Volt Output < 3.4496 V
# // Low Volt Level 2, when APS Volt Output < 3.3992 V, then this flag is SET (0x01)
# // Flag will reset once battery volt is charged above Low Volt Level 1
# // Note: now AXP192 have the Shutdown Voltage of 3.0V (B100) Def in REG 31H
def get_warning_level()
return self.read12(0x47) & 1
end
#- trigger a read every second -#
# def every_second()
# if !self.wire return nil end #- exit if not initialized -#
# end
#- display sensor value in the web UI -#
def web_sensor()
if !self.wire return nil end #- exit if not initialized -#
import string
var msg = string.format(
"{s}VBus Voltage{m}%.3f V{e}"..
"{s}VBus Current{m}%.1f mA{e}"..
"{s}Batt Voltage{m}%.3f V{e}"..
"{s}Batt Current{m}%.1f mA{e}"..
#"{s}Batt Power{m}%.3f{e}"..
"{s}Temp AXP{m}%.1f °C{e}",
self.get_vbus_voltage(), self.get_vbus_voltage(),
self.get_bat_voltage(), self.get_bat_current(),
#self.get_bat_power(),
self.get_temp()
)
tasmota.web_send_decimal(msg)
end
#- add sensor value to teleperiod -#
def json_append()
if !self.wire return nil end #- exit if not initialized -#
# import string
# var ax = int(self.accel[0] * 1000)
# var ay = int(self.accel[1] * 1000)
# var az = int(self.accel[2] * 1000)
# var msg = string.format(",\"MPU6886\":{\"AX\":%i,\"AY\":%i,\"AZ\":%i,\"GX\":%i,\"GY\":%i,\"GZ\":%i}",
# ax, ay, az, self.gyro[0], self.gyro[1], self.gyro[2])
# tasmota.response_append(msg)
end
end
#-------------------------------------------------------------
- Specialized driver for AXP192 of M5Stack Core 2
-------------------------------------------------------------#
class AXP192_M5Stack_Core2 : AXP192
def init()
super(self).init()
if self.wire
# Disable vbus hold limit
self.write1(0x30, self.read8(0x30) & 0x04 | 0x02)
# AXP192 GPIO1:OD OUTPUT
self.write1(0x92, self.read8(0x92) & 0xF8)
# AXP192 GPIO2:OD OUTPUT
self.write1(0x93, self.read8(0x93) & 0xF8)
# AXP192 RTC CHG rtc battery charging enabled
self.write1(0x35, self.read8(0x35) & 0x1C | 0xA2)
# esp32 power voltage was set to 3.35v
self.set_esp_voltage(3350)
# lcd backlight voltage was set to 2.80v
self.set_lcd_voltage(2800)
# lcd logic and sdcard voltage preset to 3.3v
self.set_ldo_voltage(2, 3300)
# vibrator voltage preset to 2v
self.set_ldo_voltage(3, 2000)
# turn on logic and sdcard
self.set_ldo_enable(2, true)
# turn on lcd backlight
self.set_dcdc_enable(3, true)
# turn green led on
self.set_led(true)
# set charging current to 100mA
self.set_chg_current(0) # 0 = 100mA
# AXP192 GPIO4 - TBC what this does really?
self.write1(0x95, self.read8(0X95) & 0x72 | 0x84)
# Button parameter setting
# Current:
# Boot time = 512ms
# Long key press time setting = 1s
# Automatic shutdown function setting when the key duration is longer than the shutdown duration = turn on
# PWROK signal delay after power on = 64ms
# Shutdown duration setting = 4s
self.write1(0x34, 0x4C)
# ADC all-on
# Bit 7: Battery voltage ADC enable
# Bit 6: Battery current ADC enable
# Bit 5: ACIN voltage ADC enable
# Bit 4: ACIN current ADC enable
# Bit 3: VBUS voltage ADC enable
# Bit 2: VBUS current ADC enable
# Bit 1: APS voltage ADC enable
# Bit 0: TS pin ADC function enable
self.write1(0x82, 0xFF)
# Reset LCD Controller
self.set_lcd_reset(false)
self.get_tasmota().delay(100) # wait for 100ms
self.set_lcd_reset(true)
self.get_tasmota().delay(100) # wait for 100ms
# bus power mode_output
self.set_buf_power_mode(false)
end
end
# set ESP voltage on DCDC1
def set_esp_voltage(voltage)
if voltage >= 3000 && voltage <= 3400
self.set_dc_voltage(1, voltage)
end
end
# set LCD backlight voltage on DCDC3
def set_lcd_voltage(voltage)
if voltage < 2500 voltage = 2500 end
if voltage > 3300 voltage = 3300 end
self.set_ldo_voltage(2, voltage)
end
# set state of the green led, GPIO 1
def set_led(state)
self.write_gpio(1, !state)
end
# LCD Controller Reset pin
def set_lcd_reset(state)
self.write_gpio(4, state)
end
# Speaker enable
def set_speaker_enable(state)
self.write_gpio(2, state)
end
# Bus Power Mode
def set_buf_power_mode(state)
if (state)
self.write1(0x12, self.read8(0x12) & 0xBF) # set EXTEN to disable 5v boost
self.write1(0x90, self.read8(0x90) & 0xF8 | 0x01) # set GPIO0 to float, using enternal pulldown resistor to enable supply from BUS_5VS
else
self.write1(0x91, self.read8(0x91) & 0x0F | 0xF0)
self.write1(0x90, self.read8(0x90) & 0xF8 | 0x02) # set GPIO0 to LDO OUTPUT , pullup N_VBUSEN to disable supply from BUS_5V
self.write1(0x12, self.read8(0x12) | 0x40) # set EXTEN to enable 5v boost
end
end
end
#-------------------------------------------------------------
- Specialized driver for AXP192 of M5StickC
-------------------------------------------------------------#
class AXP192_M5StickC : AXP192
def init()
super(self).init()
if self.wire
# Disable vbus hold limit
self.write1(0x30, 0x80)
# lcd backlight voltage was set to 2.80v
self.set_lcd_voltage(2800)
# set LDO3 3.0V TFT
self.set_ldo_voltage(3, 3000)
# Set ADC sample rate to 200hz
self.write1(0x84, 0xF2)
# set charging current to 100mA
self.set_chg_current(0) # 0 = 100mA
# esp32 power voltage was set to 3.3v
self.set_esp_voltage(3300)
# PEK key parameter setting
# Current:
# Boot time setting = 512ms
# Long press time setting = 1s
# Automatic shutdown function setting when the key duration is longer than the shutdown duration = turn on
# PWROK signal delay after power on = 64ms
# Shutdown duration setting = 4s
self.write1(0x36, 0x4C)
# Mic GPIO0 - floating
self.write1(0x90, 0x07)
# enable LDO2, LDO3, DCDC1, DCDC3
self.set_ldo_enable(2, true)
self.set_ldo_enable(3, true)
self.set_dcdc_enable(1, true)
self.set_dcdc_enable(3, true)
# Set temperature protection
self.write1(0x39, 0xFC)
# Enable RTC BAT charge
self.write1(0x35, 0xA2)
# Write1Byte(0x35, 0xa2 & (disableRTC ? 0x7F : 0xFF));
# Enable bat detection
self.write1(0x32, 0x46)
# Set Power off voltage 3.0v
self.write1(0x31, self.read8(0x31) & 0xF8 | 0x04)
# ADC all-on
# Bit 7: Battery voltage ADC enable
# Bit 6: Battery current ADC enable
# Bit 5: ACIN voltage ADC enable
# Bit 4: ACIN current ADC enable
# Bit 3: VBUS voltage ADC enable
# Bit 2: VBUS current ADC enable
# Bit 1: APS voltage ADC enable
# Bit 0: TS pin ADC function enable
self.write1(0x82, 0xFF)
end
end
# set ESP voltage on DCDC1
def set_esp_voltage(voltage)
if voltage >= 3000 && voltage <= 3400
self.set_dc_voltage(1, voltage)
end
end
# set LCD backlight voltage on DCDC3
def set_lcd_voltage(voltage)
if voltage < 2500 voltage = 2500 end
if voltage > 3300 voltage = 3300 end
self.set_ldo_voltage(2, voltage)
end
# LCD Controller Reset pin
def set_lcd_reset(state)
self.set_ldo_enable(3, state)
end
end
# axp = AXP192_M5Stack_Core2()
# tasmota.add_driver(axp)
axp = AXP192_M5StickC()
tasmota.add_driver(axp)