From d1a19119f2fc63555a7547091e1aa065f5f94fd8 Mon Sep 17 00:00:00 2001 From: Yevgeniy Kuksenko <2882631+ykuksenko@users.noreply.github.com> Date: Sat, 28 Dec 2024 17:17:39 -0600 Subject: [PATCH] EcoFlow River 3+ and Delta 3+ support Basic status works (charging, discharging, online, onbattery, ...). Charge level works. Discharge level from app shows correctly. Any sort of control does not seem to work for load or beeper/light. See code comments for details. Signed-off-by: Yevgeniy Kuksenko <2882631+ykuksenko@users.noreply.github.com> --- drivers/Makefile.am | 6 +- drivers/ecoflow-hid.c | 208 ++++++++++++++++++++++++++++++ drivers/ecoflow-hid.h | 30 +++++ drivers/usbhid-ups.c | 2 + scripts/upower/95-upower-hid.hwdb | 5 + 5 files changed, 248 insertions(+), 3 deletions(-) create mode 100644 drivers/ecoflow-hid.c create mode 100644 drivers/ecoflow-hid.h diff --git a/drivers/Makefile.am b/drivers/Makefile.am index 4ea7ab4a44..0277e93e1f 100644 --- a/drivers/Makefile.am +++ b/drivers/Makefile.am @@ -230,8 +230,8 @@ endif if WITH_LIBUSB_1_0 LIBUSB_IMPL = libusb1.c endif -USBHID_UPS_SUBDRIVERS = apc-hid.c arduino-hid.c belkin-hid.c cps-hid.c explore-hid.c \ - liebert-hid.c mge-hid.c powercom-hid.c tripplite-hid.c idowell-hid.c \ +USBHID_UPS_SUBDRIVERS = apc-hid.c arduino-hid.c belkin-hid.c cps-hid.c ecoflow-hid.c \ + explore-hid.c liebert-hid.c mge-hid.c powercom-hid.c tripplite-hid.c idowell-hid.c \ openups-hid.c powervar-hid.c delta_ups-hid.c ever-hid.c legrand-hid.c salicru-hid.c usbhid_ups_SOURCES = usbhid-ups.c libhid.c $(LIBUSB_IMPL) hidparser.c \ usb-common.c $(USBHID_UPS_SUBDRIVERS) @@ -395,7 +395,7 @@ nutdrv_qx_SOURCES += $(NUTDRV_QX_SUBDRIVERS) dist_noinst_HEADERS = \ apc_modbus.h apc-mib.h apc-iem-mib.h apc-hid.h arduino-hid.h baytech-mib.h bcmxcp.h bcmxcp_ser.h \ bcmxcp_io.h belkin.h belkin-hid.h bestpower-mib.h blazer.h cps-hid.h dstate.h \ - dummy-ups.h explore-hid.h gamatronic.h genericups.h \ + dummy-ups.h ecoflow-hid.h explore-hid.h gamatronic.h genericups.h \ generic_gpio_common.h generic_gpio_libgpiod.h \ hidparser.h hidtypes.h ietf-mib.h libhid.h libshut.h nut_libusb.h liebert-hid.h \ main.h mge-hid.h mge-mib.h mge-utalk.h \ diff --git a/drivers/ecoflow-hid.c b/drivers/ecoflow-hid.c new file mode 100644 index 0000000000..c5c107c812 --- /dev/null +++ b/drivers/ecoflow-hid.c @@ -0,0 +1,208 @@ +/* ecoflow-hid.c - subdriver to monitor EcoFlow USB/HID devices with NUT + * + * Copyright (C) + * 2003 - 2012 Arnaud Quette + * 2005 - 2006 Peter Selinger + * 2008 - 2009 Arjen de Korte + * 2013 Charles Lepple + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#include "usbhid-ups.h" +#include "ecoflow-hid.h" +#include "main.h" /* for getval() */ +#include "usb-common.h" + +#define ECOFLOW_HID_VERSION "EcoFlow HID 0.1" +/* FIXME: experimental flag to be put in upsdrv_info */ + +/* EcoFlow */ +#define ECOFLOW_VENDORID 0x3746 + +/* USB IDs device table */ +static usb_device_id_t ecoflow_usb_device_table[] = { + /* EcoFlow */ + { USB_DEVICE(ECOFLOW_VENDORID, 0xffff), NULL }, + + /* Terminating entry */ + { 0, 0, NULL } +}; + + +/* --------------------------------------------------------------- */ +/* Vendor-specific usage table */ +/* --------------------------------------------------------------- */ + +/* ECOFLOW usage table */ +static usage_lkp_t ecoflow_usage_lkp[] = { + { NULL, 0 } +}; + +static usage_tables_t ecoflow_utab[] = { + ecoflow_usage_lkp, + hid_usage_lkp, + NULL, +}; + +/* --------------------------------------------------------------- */ +/* HID2NUT lookup table */ +/* --------------------------------------------------------------- */ + +static hid_info_t ecoflow_hid2nut[] = { +/* Please revise values discovered by data walk for mappings to + * docs/nut-names.txt and group the rest under the ifdef below: + */ + /* AudibleAlarmControl: + * Delta 3+ has beeper. River 3+ does not have a beeper, only a light. + * - Both values stay enabled even when disabled in app. + * - Delta 3+ does not beep on power outage. + * - River 3+ turns its light on on power outage. Can be suppressed in mobile app. + * - Does not seem controllable. How should this be controlled? + * Attempted Commands: `bin/upscmd -u admin -p admin ecoflow beeper.(toggle|enable|disable)` + * { "ups.beeper.status", 0, 0, "UPS.PowerSummary.AudibleAlarmControl", NULL, "%s", HU_FLAG_QUICK_POLL, beeper_info }, + */ + /* These do not seem to work on both River 3+ and Delta 3+. + * { "beeper.enable", 0, 0, "UPS.PowerSummary.AudibleAlarmControl", NULL, "2", HU_TYPE_CMD, NULL }, + * { "beeper.disable", 0, 0, "UPS.PowerSummary.AudibleAlarmControl", NULL, "3", HU_TYPE_CMD, NULL }, + * { "beeper.toggle", 0, 0, "UPS.PowerSummary.AudibleAlarmControl", NULL, "0", HU_TYPE_CMD, NULL }, + * { "shutdown.reboot", 0, 0, "UPS.PowerSummary.DelayBeforeReboot", NULL, "1", HU_TYPE_CMD, NULL }, + * - FIXME: Tested command `bin/upscmd -u admin -p admin ecoflow shutdown.reboot 1` + * { "shutdown.stop", 0, 0, "UPS.PowerSummary.DelayBeforeShutdown", NULL, "1", HU_TYPE_CMD, NULL }, + * - FIXME: Tested command `bin/upscmd -u admin -p admin ecoflow shutdown.stop 1` + */ + + /* DesignCapacity: unit %, on both River 3+ and Delta 3+. */ + { "battery.capacity.nominal", 0, 0, "UPS.PowerSummary.DesignCapacity", NULL, "%.0f", 0, NULL }, + /* FullChargeCapacity: unit %, on both River 3+ and Delta 3+. */ + { "battery.capacity", 0, 0, "UPS.PowerSummary.FullChargeCapacity", NULL, "%.0f", 0, NULL }, + /* FIXME: do we want HU_FLAG_QUICK_POLL flags for these booleans? */ + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.ACPresent", NULL, NULL, HU_FLAG_QUICK_POLL, online_info}, + /* BatteryPresent: disappears when discharge limit is reached. ('ups.status: ALARM OB RB' and 'ups.alarm: No battery installed!') */ + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.BatteryPresent", NULL, NULL, HU_FLAG_QUICK_POLL, nobattery_info }, + /* BelowRemainingCapacityLimit: did not trigger when when charge was less than discharge limit. Seemed same when at %0 charge */ + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.BelowRemainingCapacityLimit", NULL, NULL, HU_FLAG_QUICK_POLL, lowbatt_info }, + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.Charging", NULL, NULL, HU_FLAG_QUICK_POLL, charging_info}, + /* CommunicationLost: Could not find how to trigger */ + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.CommunicationLost", NULL, NULL, HU_FLAG_QUICK_POLL, commfault_info }, + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.Discharging", NULL, NULL, HU_FLAG_QUICK_POLL, discharging_info}, + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.FullyCharged", NULL, NULL, HU_FLAG_QUICK_POLL, fullycharged_info }, + /* FullyDischarged: activates at APP's discharge limit */ + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.FullyDischarged", NULL, NULL, HU_FLAG_QUICK_POLL, depleted_info }, + /* NeedReplacement: activates at APP's discharge limit */ + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.NeedReplacement", NULL, NULL, HU_FLAG_QUICK_POLL, replacebatt_info}, + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.Overload", NULL, NULL, HU_FLAG_QUICK_POLL, overload_info }, + /* RemainingTimeLimitExpired: not sure how this works */ + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.RemainingTimeLimitExpired", NULL, NULL, HU_FLAG_QUICK_POLL, timelimitexpired_info }, + /* ShutdownImminent: did not trigger at discharge limit */ + { "BOOL", 0, 0, "UPS.PowerSummary.PresentStatus.ShutdownImminent", NULL, NULL, HU_FLAG_QUICK_POLL, shutdownimm_info }, + /* RemainingCapacity: unit %, load is turned off at discharge limit +1% (may be due to slow update rate) */ + { "battery.charge", 0, 0, "UPS.PowerSummary.RemainingCapacity", NULL, "%.0f", 0, NULL }, + /* RemainingCapacityLimit: corresponds to the app's discharge limit setting in % + * FIXME: not sure if battery.charge.low is the best mapping but the data is wanted. Load turns off at this charge level. + * - may want HU_FLAG_SEMI_STATIC flag. */ + { "battery.charge.low", 0, 0, "UPS.PowerSummary.RemainingCapacityLimit", NULL, "%.0f", 0, NULL }, + /* RunTimeToEmpty: unit minutes - this estimate is for full capacity, not discharge limit capacity + * FIXME: convert to seconds? */ + { "battery.runtime", 0, 0, "UPS.PowerSummary.RunTimeToEmpty", NULL, "%.0f", HU_FLAG_QUICK_POLL, NULL }, + + + /* RemainingTimeLimit: seems to be a fixed value of 120 for Delta3+ and 600 for River3+. unit minutes + * { "battery.runtime.low", 0, 0, "UPS.PowerSummary.RemainingTimeLimit", NULL, "%.0f", HU_FLAG_SEMI_STATIC, NULL }, + * WarningCapacityLimit: seems to be a fixed value at 5% for Delta3+ and 10% for River3+ + * { "battery.charge.warning", 0, 0, "UPS.PowerSummary.WarningCapacityLimit", NULL, "%.0f", HU_FLAG_SEMI_STATIC, NULL }, */ + +#if WITH_UNMAPPED_DATA_POINTS + /* ConfigActivePower: seems to be close to Wh ratings of the Devices (260Wh for River3+(286Wh are specs)/1024Wh for Delta3+(1024Wh are specs)) */ + { "unmapped.ups.flow.[4].configactivepower", 0, 0, "UPS.Flow.[4].ConfigActivePower", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.audiblealarmcontrol", 0, 0, "UPS.PowerSummary.AudibleAlarmControl", NULL, "%.0f", 0, NULL }, + /* AverageTimeToEmpty: not sure what to map this to and the values do not make sense at this point, they update way too slowly. */ + { "unmapped.ups.powersummary.averagetimetoempty", 0, 0, "UPS.PowerSummary.AverageTimeToEmpty", NULL, "%.0f", 0, NULL }, + /* AverageTimeToFull: not sure what to map this to and the values do not make sense at this point, they update way too slowly. */ + { "unmapped.ups.powersummary.averagetimetofull", 0, 0, "UPS.PowerSummary.AverageTimeToFull", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.capacitygranularity1", 0, 0, "UPS.PowerSummary.CapacityGranularity1", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.capacitygranularity2", 0, 0, "UPS.PowerSummary.CapacityGranularity2", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.capacitymode", 0, 0, "UPS.PowerSummary.CapacityMode", NULL, "%.0f", 0, NULL }, + /* ConfigVoltage: not sure what this is, had a decimal */ + { "unmapped.ups.powersummary.configvoltage", 0, 0, "UPS.PowerSummary.ConfigVoltage", NULL, "%.1f", 0, NULL }, + /* DelayBeforeReboot: does not seem to work */ + { "unmapped.ups.powersummary.delaybeforereboot", 0, 0, "UPS.PowerSummary.DelayBeforeReboot", NULL, "%.0f", 0, NULL }, + /* DelayBeforeShutdown: does not seem to work */ + { "unmapped.ups.powersummary.delaybeforeshutdown", 0, 0, "UPS.PowerSummary.DelayBeforeShutdown", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.manufacturerdate", 0, 0, "UPS.PowerSummary.ManufacturerDate", NULL, "%.0f", 0, NULL }, + /* ShutdownRequested: */ + { "unmapped.ups.powersummary.presentstatus.shutdownrequested", 0, 0, "UPS.PowerSummary.PresentStatus.ShutdownRequested", NULL, "%.0f", 0, NULL }, + /* VoltageNotRegulated: not sure how to trigger this */ + { "unmapped.ups.powersummary.presentstatus.voltagenotregulated", 0, 0, "UPS.PowerSummary.PresentStatus.VoltageNotRegulated", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.rechargeable", 0, 0, "UPS.PowerSummary.Rechargeable", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.remainingtimelimit", 0, 0, "UPS.PowerSummary.RemainingTimeLimit", NULL, "%.0f", 0, NULL }, + /* Voltage: Probably should have a decimal to match ConfigVoltage, does not seem to change even at 0% capacity */ + { "unmapped.ups.powersummary.voltage", 0, 0, "UPS.PowerSummary.Voltage", NULL, "%.1f", 0, NULL }, + { "unmapped.ups.powersummary.warningcapacitylimit", 0, 0, "UPS.PowerSummary.WarningCapacityLimit", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.idevicechemistry", 0, 0, "UPS.PowerSummary.iDeviceChemistry", NULL, "%.0f", 0, NULL }, + { "unmapped.ups.powersummary.ioeminformation", 0, 0, "UPS.PowerSummary.iOEMInformation", NULL, "%.0f", 0, NULL }, +#endif /* if WITH_UNMAPPED_DATA_POINTS */ + + /* end of structure. */ + { NULL, 0, 0, NULL, NULL, NULL, 0, NULL } +}; + +static const char *ecoflow_format_model(HIDDevice_t *hd) { + return hd->Product; +} + +static const char *ecoflow_format_mfr(HIDDevice_t *hd) { + return hd->Vendor ? hd->Vendor : "EcoFlow"; +} + +static const char *ecoflow_format_serial(HIDDevice_t *hd) { + return hd->Serial; +} + +/* this function allows the subdriver to "claim" a device: return 1 if + * the device is supported by this subdriver, else 0. */ +static int ecoflow_claim(HIDDevice_t *hd) +{ + int status = is_usb_device_supported(ecoflow_usb_device_table, hd); + + switch (status) + { + case POSSIBLY_SUPPORTED: + /* by default, reject, unless the productid option is given */ + if (getval("productid")) { + return 1; + } + possibly_supported("EcoFlow", hd); + return 0; + + case SUPPORTED: + return 1; + + case NOT_SUPPORTED: + default: + return 0; + } +} + +subdriver_t ecoflow_subdriver = { + ECOFLOW_HID_VERSION, + ecoflow_claim, + ecoflow_utab, + ecoflow_hid2nut, + ecoflow_format_model, + ecoflow_format_mfr, + ecoflow_format_serial, + fix_report_desc, /* may optionally be customized, see cps-hid.c for example */ +}; diff --git a/drivers/ecoflow-hid.h b/drivers/ecoflow-hid.h new file mode 100644 index 0000000000..91fc828b61 --- /dev/null +++ b/drivers/ecoflow-hid.h @@ -0,0 +1,30 @@ +/* ecoflow-hid.h - subdriver to monitor EcoFlow USB/HID devices with NUT + * + * Copyright (C) + * 2003 - 2009 Arnaud Quette + * 2005 - 2006 Peter Selinger + * 2008 - 2009 Arjen de Korte + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA + */ + +#ifndef ECOFLOW_HID_H +#define ECOFLOW_HID_H + +#include "usbhid-ups.h" + +extern subdriver_t ecoflow_subdriver; + +#endif /* ECOFLOW_HID_H */ diff --git a/drivers/usbhid-ups.c b/drivers/usbhid-ups.c index 539f52c7e5..cfb5d8bd04 100644 --- a/drivers/usbhid-ups.c +++ b/drivers/usbhid-ups.c @@ -55,6 +55,7 @@ # include "belkin-hid.h" # include "cps-hid.h" # include "delta_ups-hid.h" +# include "ecoflow-hid.h" # include "ever-hid.h" # include "idowell-hid.h" # include "legrand-hid.h" @@ -79,6 +80,7 @@ static subdriver_t *subdriver_list[] = { &belkin_subdriver, &cps_subdriver, &delta_ups_subdriver, + &ecoflow_subdriver, &ever_subdriver, &idowell_subdriver, &legrand_subdriver, diff --git a/scripts/upower/95-upower-hid.hwdb b/scripts/upower/95-upower-hid.hwdb index 7e81fddf5b..a45706a78e 100644 --- a/scripts/upower/95-upower-hid.hwdb +++ b/scripts/upower/95-upower-hid.hwdb @@ -207,6 +207,11 @@ usb:v2E66p0302* UPOWER_BATTERY_TYPE=ups UPOWER_VENDOR=Salicru +# EcoFlow +usb:v3746pFFFF* + UPOWER_BATTERY_TYPE=ups + UPOWER_VENDOR=EcoFlow + # Powervar usb:v4234p0002* UPOWER_BATTERY_TYPE=ups