diff --git a/CODEOWNERS b/CODEOWNERS
index ace489c4b..1f67b3c34 100644
--- a/CODEOWNERS
+++ b/CODEOWNERS
@@ -158,6 +158,7 @@
/bundles/org.openhab.binding.modbus/ @ssalonen
/bundles/org.openhab.binding.modbus.e3dc/ @weymann
/bundles/org.openhab.binding.modbus.helioseasycontrols/ @bern77
+/bundles/org.openhab.binding.modbus.sbc/ @fwolter
/bundles/org.openhab.binding.modbus.stiebeleltron/ @pail23
/bundles/org.openhab.binding.modbus.studer/ @giovannimirulla
/bundles/org.openhab.binding.modbus.sunspec/ @mrbig
diff --git a/bom/openhab-addons/pom.xml b/bom/openhab-addons/pom.xml
index 78f2ff07e..8569f1c6e 100644
--- a/bom/openhab-addons/pom.xml
+++ b/bom/openhab-addons/pom.xml
@@ -776,6 +776,11 @@
org.openhab.binding.modbus.helioseasycontrols
${project.version}
+
+ org.openhab.addons.bundles
+ org.openhab.binding.modbus.sbc
+ ${project.version}
+
org.openhab.addons.bundles
org.openhab.binding.modbus.stiebeleltron
diff --git a/bundles/org.openhab.binding.modbus.sbc/NOTICE b/bundles/org.openhab.binding.modbus.sbc/NOTICE
new file mode 100644
index 000000000..38d625e34
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/NOTICE
@@ -0,0 +1,13 @@
+This content is produced and maintained by the openHAB project.
+
+* Project home: https://www.openhab.org
+
+== Declared Project Licenses
+
+This program and the accompanying materials are made available under the terms
+of the Eclipse Public License 2.0 which is available at
+https://www.eclipse.org/legal/epl-2.0/.
+
+== Source Code
+
+https://github.com/openhab/openhab-addons
diff --git a/bundles/org.openhab.binding.modbus.sbc/README.md b/bundles/org.openhab.binding.modbus.sbc/README.md
new file mode 100644
index 000000000..d1d75f353
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/README.md
@@ -0,0 +1,74 @@
+# Modbus Saia Burgess Controls Binding
+
+This binding interfaces the energy meter series ALD1 by Saia Burgess Controls (SBC) via Modbus.
+
+## Supported Things
+
+The following Things are supported:
+
+- `ald1Unidirectional`: 1-phase 32A one-way energy meter ALD1D5FD00A3A00
+- `ald1Bidirectional`: 1-phase 32A two-way energy meter ALD1B5FD00A3A00
+
+## Discovery
+
+This binding does not support discovery.
+
+## Thing Configuration
+
+The following configuration parameter applys to `ald1Unidirectional` and `ald1Bidirectional`.
+
+| Name | Description | Type | Required |
+|---------------|------------------------------------------|---------|----------|
+| pollInterval | Time between polling the data in ms | Integer | yes |
+
+The Thing needs a Modbus serial slave Bridge to operate.
+
+One of the following serial settings need to be configured in the Bridge:
+
+- 9600 baud, 2 stop bit, no parity
+- 9600 baud, 1 stop bit, even parity
+- 9600 baud, 1 stop bit, odd parity
+
+## Channels
+
+The following Channels apply to `ald1Unidirectional` and `ald1Bidirectional` if not stated otherwise.
+
+| Name | Type | Description |
+|---------------------|--------------------------|---------------------------------------------------------------|
+| total_energy | Number:Energy | Energy Total |
+| partial_energy | Number:Energy | Energy Counter Resettable (only unidirectional meter) |
+| feeding_back_energy | Number:Energy | Energy Feeding Back (only bidirectional meter) |
+| voltage | Number:ElectricPotential | Effective Voltage |
+| current | Number:ElectricCurrent | Effective Current |
+| active_power | Number:Power | Effective Active Power (negative numbers mean feeding back) |
+| reactive_power | Number:Power | Effective Reactive Power (negative numbers mean feeding back) |
+| power_factor | Number:Dimensionless | Power Factor |
+
+## Full Example
+
+### .items
+
+```
+Number:Energy ALD1_Total_Energy "[%.2f %unit%]" {channel="modbus:ald1Bidirectional:8b6e85623b:total_energy"}
+Number:Energy ALD1_Feeding_Back_Energy "[%.2f %unit%]" {channel="modbus:ald1Bidirectional:8b6e85623b:feeding_back_energy"}
+Number:ElectricPotential ALD1_Voltage "[%d %unit%]" {channel="modbus:ald1Bidirectional:8b6e85623b:voltage"}
+Number:ElectricCurrent ALD1_Current "[%.1f %unit%]" {channel="modbus:ald1Bidirectional:8b6e85623b:current"}
+Number:Power ALD1_Active_Power "[%.2f %unit%]" {channel="modbus:ald1Bidirectional:8b6e85623b:active_power"}
+Number:Power ALD1_Reactive_Power "[%.2f %unit%]" {channel="modbus:ald1Bidirectional:8b6e85623b:reactive_power"}
+Number:Dimensionless ALD1_Power_Factor "[%.2f]" {channel="modbus:ald1Bidirectional:8b6e85623b:power_factor"}
+```
+
+### .sitemap
+
+```
+sitemap ald1 label="ALD1 Energy Meter"
+{
+ Default item=ALD1_Total_Energy label="Total Energy"
+ Default item=ALD1_Feeding_Back_Energy label="Feeding Back Energy"
+ Default item=ALD1_Voltage label="Voltage"
+ Default item=ALD1_Current label="Current"
+ Default item=ALD1_Active_Power label="Active Power"
+ Default item=ALD1_Reactive_Power label="Reactive Power"
+ Default item=ALD1_Power_Factor label="Power Factor"
+}
+```
diff --git a/bundles/org.openhab.binding.modbus.sbc/pom.xml b/bundles/org.openhab.binding.modbus.sbc/pom.xml
new file mode 100644
index 000000000..7da99d260
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/pom.xml
@@ -0,0 +1,26 @@
+
+
+
+ 4.0.0
+
+
+ org.openhab.addons.bundles
+ org.openhab.addons.reactor.bundles
+ 3.1.0-SNAPSHOT
+
+
+ org.openhab.binding.modbus.sbc
+
+ openHAB Add-ons :: Bundles :: Modbus SBC Binding
+
+
+
+ org.openhab.addons.bundles
+ org.openhab.binding.modbus
+ ${project.version}
+ provided
+
+
+
+
diff --git a/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Configuration.java b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Configuration.java
new file mode 100644
index 000000000..d69e491e3
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Configuration.java
@@ -0,0 +1,25 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.modbus.sbc.internal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+
+/**
+ * The {@link ALD1Configuration} class contains fields mapping thing configuration parameters.
+ *
+ * @author Fabian Wolter - Initial contribution
+ */
+@NonNullByDefault
+public class ALD1Configuration {
+ public int pollInterval;
+}
diff --git a/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Handler.java b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Handler.java
new file mode 100644
index 000000000..6952cf35e
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Handler.java
@@ -0,0 +1,100 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.modbus.sbc.internal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.binding.modbus.handler.BaseModbusThingHandler;
+import org.openhab.core.io.transport.modbus.AsyncModbusFailure;
+import org.openhab.core.io.transport.modbus.AsyncModbusReadResult;
+import org.openhab.core.io.transport.modbus.ModbusBitUtilities;
+import org.openhab.core.io.transport.modbus.ModbusReadFunctionCode;
+import org.openhab.core.io.transport.modbus.ModbusReadRequestBlueprint;
+import org.openhab.core.library.types.QuantityType;
+import org.openhab.core.thing.ChannelUID;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingStatus;
+import org.openhab.core.thing.ThingStatusDetail;
+import org.openhab.core.types.Command;
+import org.openhab.core.types.RefreshType;
+
+/**
+ * The {@link ALD1Handler} is responsible for handling commands, which are
+ * sent to one of the channels.
+ *
+ * @author Fabian Wolter - Initial contribution
+ */
+@NonNullByDefault
+public class ALD1Handler extends BaseModbusThingHandler {
+ private static final int FIRST_READ_REGISTER = 28;
+ private static final int READ_LENGTH = 13;
+ private static final int TRIES = 1;
+ private ALD1Configuration config = new ALD1Configuration();
+ private @Nullable ModbusReadRequestBlueprint blueprint;
+
+ public ALD1Handler(Thing thing) {
+ super(thing);
+ }
+
+ @Override
+ public void handleCommand(ChannelUID channelUID, Command command) {
+ ModbusReadRequestBlueprint localBlueprint = blueprint;
+ if (command instanceof RefreshType && localBlueprint != null) {
+ submitOneTimePoll(localBlueprint, this::readSuccessful, this::readError);
+ }
+ }
+
+ @Override
+ public void modbusInitialize() {
+ config = getConfigAs(ALD1Configuration.class);
+
+ if (config.pollInterval <= 0) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
+ "Invalid poll interval: " + config.pollInterval);
+ return;
+ }
+
+ ModbusReadRequestBlueprint localBlueprint = blueprint = new ModbusReadRequestBlueprint(getSlaveId(),
+ ModbusReadFunctionCode.READ_MULTIPLE_REGISTERS, FIRST_READ_REGISTER - 1, READ_LENGTH, TRIES);
+
+ updateStatus(ThingStatus.UNKNOWN);
+
+ registerRegularPoll(localBlueprint, config.pollInterval, 0, this::readSuccessful, this::readError);
+ }
+
+ private void readSuccessful(AsyncModbusReadResult result) {
+ result.getRegisters().ifPresent(registers -> {
+ if (getThing().getStatus() != ThingStatus.ONLINE) {
+ updateStatus(ThingStatus.ONLINE);
+ }
+
+ for (ALD1Registers channel : ALD1Registers.values()) {
+ int index = channel.getRegisterNumber() - FIRST_READ_REGISTER;
+
+ ModbusBitUtilities.extractStateFromRegisters(registers, index, channel.getType())
+ .map(d -> d.toBigDecimal().multiply(channel.getMultiplier()))
+ .map(bigDecimal -> new QuantityType<>(bigDecimal, channel.getUnit()))
+ .ifPresent(v -> updateState(createChannelUid(channel), v));
+ }
+ });
+ }
+
+ private void readError(AsyncModbusFailure error) {
+ updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
+ "Failed to retrieve data: " + error.getCause().getMessage());
+ }
+
+ private ChannelUID createChannelUid(ALD1Registers channel) {
+ return new ChannelUID(thing.getUID(), channel.toString().toLowerCase());
+ }
+}
diff --git a/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Registers.java b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Registers.java
new file mode 100644
index 000000000..077b6eb4c
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/ALD1Registers.java
@@ -0,0 +1,70 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.modbus.sbc.internal;
+
+import static org.openhab.core.io.transport.modbus.ModbusConstants.ValueType.*;
+
+import java.math.BigDecimal;
+
+import javax.measure.Unit;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.io.transport.modbus.ModbusConstants;
+import org.openhab.core.io.transport.modbus.ModbusConstants.ValueType;
+import org.openhab.core.library.unit.Units;
+
+/**
+ * The {@link ALD1Registers} is responsible for defining Modbus registers and their units.
+ *
+ * @author Fabian Wolter - Initial contribution
+ */
+@NonNullByDefault
+public enum ALD1Registers {
+ // the following register numbers are 1-based. They need to be converted before sending them on the wire.
+ TOTAL_ENERGY(0.01f, 28, UINT32, Units.KILOWATT_HOUR),
+ PARTIAL_ENERGY(0.01f, 30, UINT32, Units.KILOWATT_HOUR), // only unidirectional meters
+ FEEDING_BACK_ENERGY(0.01f, 30, UINT32, Units.KILOWATT_HOUR), // only bidirectional meters
+ VOLTAGE(1, 36, UINT16, Units.VOLT),
+ CURRENT(0.1f, 37, UINT16, Units.AMPERE),
+ ACTIVE_POWER(10, 38, INT16, Units.WATT),
+ REACTIVE_POWER(10, 39, INT16, Units.VAR),
+ POWER_FACTOR(0.01f, 40, INT16, Units.ONE);
+
+ private BigDecimal multiplier;
+ private int registerNumber;
+ private ModbusConstants.ValueType type;
+ private Unit> unit;
+
+ private ALD1Registers(float multiplier, int registerNumber, ValueType type, Unit> unit) {
+ this.multiplier = new BigDecimal(multiplier);
+ this.registerNumber = registerNumber;
+ this.type = type;
+ this.unit = unit;
+ }
+
+ public Unit> getUnit() {
+ return unit;
+ }
+
+ public BigDecimal getMultiplier() {
+ return multiplier;
+ }
+
+ public int getRegisterNumber() {
+ return registerNumber;
+ }
+
+ public ModbusConstants.ValueType getType() {
+ return type;
+ }
+}
diff --git a/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/SBCBindingConstants.java b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/SBCBindingConstants.java
new file mode 100644
index 000000000..8b00a41ef
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/SBCBindingConstants.java
@@ -0,0 +1,31 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.modbus.sbc.internal;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.binding.modbus.ModbusBindingConstants;
+import org.openhab.core.thing.ThingTypeUID;
+
+/**
+ * The {@link SBCBindingConstants} class defines common constants, which are
+ * used across the whole binding.
+ *
+ * @author Fabian Wolter - Initial contribution
+ */
+@NonNullByDefault
+public class SBCBindingConstants {
+ public static final ThingTypeUID THING_TYPE_ALD1_UNIDIRECTIONAL = new ThingTypeUID(
+ ModbusBindingConstants.BINDING_ID, "ald1Unidirectional");
+ public static final ThingTypeUID THING_TYPE_ALD1_BIDIRECTIONAL = new ThingTypeUID(ModbusBindingConstants.BINDING_ID,
+ "ald1Bidirectional");
+}
diff --git a/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/SBCHandlerFactory.java b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/SBCHandlerFactory.java
new file mode 100644
index 000000000..a7410861a
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/src/main/java/org/openhab/binding/modbus/sbc/internal/SBCHandlerFactory.java
@@ -0,0 +1,56 @@
+/**
+ * Copyright (c) 2010-2021 Contributors to the openHAB project
+ *
+ * See the NOTICE file(s) distributed with this work for additional
+ * information.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License 2.0 which is available at
+ * http://www.eclipse.org/legal/epl-2.0
+ *
+ * SPDX-License-Identifier: EPL-2.0
+ */
+package org.openhab.binding.modbus.sbc.internal;
+
+import static org.openhab.binding.modbus.sbc.internal.SBCBindingConstants.*;
+
+import java.util.Set;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
+import org.openhab.core.thing.Thing;
+import org.openhab.core.thing.ThingTypeUID;
+import org.openhab.core.thing.binding.BaseThingHandlerFactory;
+import org.openhab.core.thing.binding.ThingHandler;
+import org.openhab.core.thing.binding.ThingHandlerFactory;
+import org.osgi.service.component.annotations.Component;
+
+/**
+ * The {@link SBCHandlerFactory} is responsible for creating things and thing
+ * handlers.
+ *
+ * @author Fabian Wolter - Initial contribution
+ */
+@NonNullByDefault
+@Component(configurationPid = "binding.sbc", service = ThingHandlerFactory.class)
+public class SBCHandlerFactory extends BaseThingHandlerFactory {
+
+ private static final Set SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_ALD1_UNIDIRECTIONAL,
+ THING_TYPE_ALD1_BIDIRECTIONAL);
+
+ @Override
+ public boolean supportsThingType(ThingTypeUID thingTypeUID) {
+ return SUPPORTED_THING_TYPES_UIDS.contains(thingTypeUID);
+ }
+
+ @Override
+ protected @Nullable ThingHandler createHandler(Thing thing) {
+ ThingTypeUID thingTypeUID = thing.getThingTypeUID();
+
+ if (THING_TYPE_ALD1_UNIDIRECTIONAL.equals(thingTypeUID) || THING_TYPE_ALD1_BIDIRECTIONAL.equals(thingTypeUID)) {
+ return new ALD1Handler(thing);
+ }
+
+ return null;
+ }
+}
diff --git a/bundles/org.openhab.binding.modbus.sbc/src/main/resources/OH-INF/config/config.xml b/bundles/org.openhab.binding.modbus.sbc/src/main/resources/OH-INF/config/config.xml
new file mode 100644
index 000000000..001cf9136
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/src/main/resources/OH-INF/config/config.xml
@@ -0,0 +1,15 @@
+
+
+
+
+
+
+ Time between polling the data in ms.
+ 5000
+
+
+
+
diff --git a/bundles/org.openhab.binding.modbus.sbc/src/main/resources/OH-INF/thing/thing-types.xml b/bundles/org.openhab.binding.modbus.sbc/src/main/resources/OH-INF/thing/thing-types.xml
new file mode 100644
index 000000000..d62cd11cb
--- /dev/null
+++ b/bundles/org.openhab.binding.modbus.sbc/src/main/resources/OH-INF/thing/thing-types.xml
@@ -0,0 +1,100 @@
+
+
+
+
+
+
+
+
+
+
+ Unidirectional one-phase energy meter connected via Modbus.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Bidirectional one-phase energy meter connected via Modbus.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Number:Energy
+
+
+
+
+
+ Number:Energy
+
+
+
+
+
+ Number:Energy
+
+
+
+
+
+ Number:ElectricPotential
+
+
+
+
+
+ Number:ElectricCurrent
+
+
+
+
+
+ Number:Power
+
+ Negative numbers mean feeding back
+
+
+
+
+ Number:Power
+
+ Negative numbers mean feeding back
+
+
+
+
+ Number:Dimensionless
+
+
+
+
diff --git a/bundles/pom.xml b/bundles/pom.xml
index 8915314b7..9722fb7f9 100644
--- a/bundles/pom.xml
+++ b/bundles/pom.xml
@@ -188,6 +188,7 @@
org.openhab.binding.minecraft
org.openhab.binding.modbus
org.openhab.binding.modbus.e3dc
+ org.openhab.binding.modbus.sbc
org.openhab.binding.modbus.studer
org.openhab.binding.modbus.sunspec
org.openhab.binding.modbus.stiebeleltron
diff --git a/features/openhab-addons/src/main/resources/footer.xml b/features/openhab-addons/src/main/resources/footer.xml
index 291d05365..d38fbdf3f 100644
--- a/features/openhab-addons/src/main/resources/footer.xml
+++ b/features/openhab-addons/src/main/resources/footer.xml
@@ -31,6 +31,7 @@
mvn:org.openhab.addons.bundles/org.openhab.binding.modbus/${project.version}
mvn:org.openhab.addons.bundles/org.openhab.binding.modbus.e3dc/${project.version}
mvn:org.openhab.addons.bundles/org.openhab.binding.modbus.helioseasycontrols/${project.version}
+ mvn:org.openhab.addons.bundles/org.openhab.binding.modbus.sbc/${project.version}
mvn:org.openhab.addons.bundles/org.openhab.binding.modbus.stiebeleltron/${project.version}
mvn:org.openhab.addons.bundles/org.openhab.binding.modbus.studer/${project.version}
mvn:org.openhab.addons.bundles/org.openhab.binding.modbus.sunspec/${project.version}