Skip to content

Commit

Permalink
Victron: add EVCS (#10959)
Browse files Browse the repository at this point in the history
Co-authored-by: Philipp Trenz <mail@philipptrenz.de>
  • Loading branch information
andig and philipptrenz authored Dec 1, 2023
1 parent cce46f1 commit 82ecb9f
Show file tree
Hide file tree
Showing 2 changed files with 79 additions and 21 deletions.
83 changes: 62 additions & 21 deletions charger/victron.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,41 +26,77 @@ import (
"github.com/evcc-io/evcc/util"
"github.com/evcc-io/evcc/util/modbus"
"github.com/evcc-io/evcc/util/sponsor"
"github.com/spf13/cast"
)

// Victron charger implementation
type Victron struct {
conn *modbus.Connection
regs victronRegs
}

const (
victronRegMode = 3815
victronRegEnergy = 3816
victronRegPower = 3821
victronRegStatus = 3824
victronRegSetCurrent = 3825
victronRegEnabled = 3826
type victronRegs struct {
Mode uint16
Energy uint16
Power uint16
Status uint16
SetCurrent uint16
Enabled uint16
isGX bool
}

var (
victronGX = victronRegs{
Mode: 3815,
Energy: 3816,
Power: 3821,
Status: 3824,
SetCurrent: 3825,
Enabled: 3826,
isGX: true,
}

victronEVCS = victronRegs{
Mode: 5009,
Energy: 5021,
Power: 5014,
Status: 5015,
SetCurrent: 5016,
Enabled: 5010,
isGX: false,
}
)

func init() {
registry.Add("victron", NewVictronFromConfig)
registry.Add("victron", NewVictronGXFromConfig)
registry.Add("victron-evcs", NewVictronEVCSFromConfig)
}

// NewVictronGXFromConfig creates a ABB charger from generic config
func NewVictronGXFromConfig(other map[string]interface{}) (api.Charger, error) {
return NewVictronFromConfig(other, victronGX)
}

// NewVictronEVCSFromConfig creates a ABB charger from generic config
func NewVictronEVCSFromConfig(other map[string]interface{}) (api.Charger, error) {
return NewVictronFromConfig(other, victronEVCS)
}

// NewVictronFromConfig creates a ABB charger from generic config
func NewVictronFromConfig(other map[string]interface{}) (api.Charger, error) {
func NewVictronFromConfig(other map[string]interface{}, regs victronRegs) (api.Charger, error) {
cc := modbus.TcpSettings{
ID: 100,
ID: cast.ToUint8(regs.isGX) * 100,
}

if err := util.DecodeOther(other, &cc); err != nil {
return nil, err
}

return NewVictron(cc.URI, cc.ID)
return NewVictron(cc.URI, cc.ID, victronEVCS)
}

// NewVictron creates Victron charger
func NewVictron(uri string, slaveID uint8) (api.Charger, error) {
func NewVictron(uri string, slaveID uint8, regs victronRegs) (api.Charger, error) {
conn, err := modbus.NewConnection(uri, "", "", 0, modbus.Tcp, slaveID)
if err != nil {
return nil, err
Expand All @@ -75,9 +111,10 @@ func NewVictron(uri string, slaveID uint8) (api.Charger, error) {

wb := &Victron{
conn: conn,
regs: regs,
}

b, err := wb.conn.ReadHoldingRegisters(victronRegMode, 1)
b, err := wb.conn.ReadHoldingRegisters(wb.regs.Mode, 1)
if err != nil {
return nil, err
}
Expand All @@ -91,7 +128,7 @@ func NewVictron(uri string, slaveID uint8) (api.Charger, error) {

// Status implements the api.Charger interface
func (wb *Victron) Status() (api.ChargeStatus, error) {
b, err := wb.conn.ReadHoldingRegisters(victronRegStatus, 1)
b, err := wb.conn.ReadHoldingRegisters(wb.regs.Status, 1)
if err != nil {
return api.StatusNone, err
}
Expand All @@ -100,7 +137,7 @@ func (wb *Victron) Status() (api.ChargeStatus, error) {
switch u {
case 0, 1, 2, 3:
return api.ChargeStatusString(string('A' + rune(binary.BigEndian.Uint16(b))))
case 5, 6:
case 5, 6, 21:
return api.StatusB, nil
default:
return api.StatusNone, fmt.Errorf("invalid status: %d", u)
Expand All @@ -109,7 +146,7 @@ func (wb *Victron) Status() (api.ChargeStatus, error) {

// Enabled implements the api.Charger interface
func (wb *Victron) Enabled() (bool, error) {
b, err := wb.conn.ReadHoldingRegisters(victronRegEnabled, 1)
b, err := wb.conn.ReadHoldingRegisters(wb.regs.Enabled, 1)
if err != nil {
return false, err
}
Expand All @@ -124,21 +161,21 @@ func (wb *Victron) Enable(enable bool) error {
u = 1
}

_, err := wb.conn.WriteSingleRegister(victronRegEnabled, u)
_, err := wb.conn.WriteSingleRegister(wb.regs.Enabled, u)
return err
}

// MaxCurrent implements the api.Charger interface
func (wb *Victron) MaxCurrent(current int64) error {
_, err := wb.conn.WriteSingleRegister(victronRegSetCurrent, uint16(current))
_, err := wb.conn.WriteSingleRegister(wb.regs.SetCurrent, uint16(current))
return err
}

var _ api.Meter = (*Victron)(nil)

// CurrentPower implements the api.Meter interface
func (wb *Victron) CurrentPower() (float64, error) {
b, err := wb.conn.ReadHoldingRegisters(victronRegPower, 1)
b, err := wb.conn.ReadHoldingRegisters(wb.regs.Power, 1)
if err != nil {
return 0, err
}
Expand All @@ -150,10 +187,14 @@ var _ api.ChargeRater = (*Victron)(nil)

// ChargedEnergy implements the api.MeterEnergy interface
func (wb *Victron) ChargedEnergy() (float64, error) {
b, err := wb.conn.ReadHoldingRegisters(victronRegEnergy, 2)
b, err := wb.conn.ReadHoldingRegisters(wb.regs.Energy, cast.ToUint16(wb.regs.isGX))
if err != nil {
return 0, err
}

return float64(binary.BigEndian.Uint32(b)) / 100, nil
if wb.regs.isGX {
return float64(binary.BigEndian.Uint32(b)) / 100, nil
}

return float64(binary.BigEndian.Uint16(b)) / 100, nil
}
17 changes: 17 additions & 0 deletions templates/definition/charger/victron-evcs.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
template: victron-evcs
products:
- brand: Victron
description:
generic: EV charging station
requirements:
evcc: ["sponsorship"]
description:
en: Enter the host of the charger (not the GX device) and ensure that the charger is in manual mode.
de: Trage den Host der Wallbox (nicht des GX-Geräts) ein und stelle sicher, dass die Wallbox sich im Modus "Manual" befindet.
params:
- name: modbus
choice: ["tcpip"]
id: 1
render: |
type: victron-evcs
{{- include "modbus" . }}

0 comments on commit 82ecb9f

Please sign in to comment.