-
-
Notifications
You must be signed in to change notification settings - Fork 359
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
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 <[email protected]>
- Loading branch information
Showing
5 changed files
with
248 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,208 @@ | ||
/* ecoflow-hid.c - subdriver to monitor EcoFlow USB/HID devices with NUT | ||
* | ||
* Copyright (C) | ||
* 2003 - 2012 Arnaud Quette <[email protected]> | ||
* 2005 - 2006 Peter Selinger <[email protected]> | ||
* 2008 - 2009 Arjen de Korte <[email protected]> | ||
* 2013 Charles Lepple <[email protected]> | ||
* | ||
* 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 */ | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,30 @@ | ||
/* ecoflow-hid.h - subdriver to monitor EcoFlow USB/HID devices with NUT | ||
* | ||
* Copyright (C) | ||
* 2003 - 2009 Arnaud Quette <[email protected]> | ||
* 2005 - 2006 Peter Selinger <[email protected]> | ||
* 2008 - 2009 Arjen de Korte <[email protected]> | ||
* | ||
* 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 */ |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters