From ccdab295e738fe53a0335719aa9c41f3be43ee5d Mon Sep 17 00:00:00 2001
From: Theo Arends <11044339+arendst@users.noreply.github.com>
Date: Sat, 4 Mar 2023 15:06:52 +0100
Subject: [PATCH] Refactor PCF8574 driver
---
.../xdrv_28_pcf8574_v1.ino | 386 ------------------
..._28_pcf8574.ino => xdrv_28_pcf8574_v2.ino} | 91 ++---
.../tasmota_xdrv_driver/xdrv_67_mcp23xxx.ino | 2 +-
3 files changed, 25 insertions(+), 454 deletions(-)
delete mode 100644 tasmota/tasmota_xdrv_driver/xdrv_28_pcf8574_v1.ino
rename tasmota/tasmota_xdrv_driver/{xdrv_28_pcf8574.ino => xdrv_28_pcf8574_v2.ino} (91%)
diff --git a/tasmota/tasmota_xdrv_driver/xdrv_28_pcf8574_v1.ino b/tasmota/tasmota_xdrv_driver/xdrv_28_pcf8574_v1.ino
deleted file mode 100644
index 841da4e70..000000000
--- a/tasmota/tasmota_xdrv_driver/xdrv_28_pcf8574_v1.ino
+++ /dev/null
@@ -1,386 +0,0 @@
-/*
- xdrv_28_pcf8574.ino - PCF8574 I2C support for Tasmota
-
- Copyright (C) 2021 Stefan Bode
-
- 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 3 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, see .
-*/
-
-#ifdef USE_I2C
-#ifdef USE_PCF8574_V1
-/*********************************************************************************************\
- * PCF8574 - I2C IO Expander
- *
- * I2C Address: PCF8574 = 0x20 .. 0x27 (0x27 is not supported),
- * PCF8574A = 0x39 .. 0x3F (0x38 is not supported)
-\*********************************************************************************************/
-
-#define XDRV_28 28
-#define XI2C_02 2 // See I2CDEVICES.md
-
-// Start address and count can be overriden in user_config_override.h to allow better
-// sharing of the I2C address space. Still the covered range must remains valid.
-// A count of 0 can be used totaly disable any of the 2 ranges.
-// By default, the following addresses are explicitly excluded (as per the docs) :
-// - 0x27 and 0x37 are reserved for USE_DISPLAY_LCD in xdsp_01_lcd.ino
-// - 0X38 is reserved for other sensors
-// If the respective drivers are not used, overrides allows to recover those addresses
-// If defined, USE_MCP230xx_ADDR is also always excluded
-
-// PCF8574 address range from 0x20 to 0x26
-#ifndef PCF8574_ADDR1
-#define PCF8574_ADDR1 0x20 // PCF8574
-#endif
-#ifndef PCF8574_ADDR1_COUNT
-#define PCF8574_ADDR1_COUNT 7
-#endif
-// PCF8574A address range from 0x39 to 0x3E
-#ifndef PCF8574_ADDR2
-#define PCF8574_ADDR2 0x39 // PCF8574A
-#endif
-#ifndef PCF8574_ADDR2_COUNT
-#define PCF8574_ADDR2_COUNT 6
-#endif
-
-// Consitency tests - Checked across the complete range for the PCF8574/PCF8574A to allow override
-#if (PCF8574_ADDR1 < 0x20) || ((PCF8574_ADDR1 + PCF8574_ADDR1_COUNT - 1) > 0x27)
-#error PCF8574_ADDR1 and/or PCF8574_ADDR1_COUNT badly overriden. Fix your user_config_override
-#endif
-#if (PCF8574_ADDR2 < 0x38) || ((PCF8574_ADDR2 + PCF8574_ADDR2_COUNT - 1) > 0x3F)
-#error PCF8574_ADDR2 and/or PCF8574_ADDR2_COUNT badly overriden. Fix your user_config_override.
-#endif
-
-struct PCF8574 {
- int error;
- uint8_t pin[64];
- uint8_t address[MAX_PCF8574];
- uint8_t pin_mask[MAX_PCF8574] = { 0 };
-#ifdef USE_PCF8574_MQTTINPUT
- uint8_t last_input[MAX_PCF8574] = { 0 };
-#endif
- uint8_t max_connected_ports = 0; // Max numbers of devices comming from PCF8574 modules
- uint8_t max_devices = 0; // Max numbers of PCF8574 modules
- char stype[9];
- bool type = false;
-} Pcf8574;
-
-uint8_t Pcf8574Read(uint8_t idx)
-{
- Wire.requestFrom(Pcf8574.address[idx],(uint8_t)1);
- return Wire.read();
-}
-
-uint8_t Pcf8574Write(uint8_t idx)
-{
- Wire.beginTransmission(Pcf8574.address[idx]);
- Wire.write(Pcf8574.pin_mask[idx]);
- return Wire.endTransmission();
-}
-
-void Pcf8574SwitchRelay(void)
-{
- for (uint32_t i = 0; i < TasmotaGlobal.devices_present; i++) {
- uint8_t relay_state = bitRead(XdrvMailbox.index, i);
-
- if (Pcf8574.max_devices > 0 && Pcf8574.pin[i] < 99) {
- uint8_t board = Pcf8574.pin[i]>>3;
- uint8_t pin = Pcf8574.pin[i]&0x7;
- uint8_t oldpinmask = Pcf8574.pin_mask[board];
- uint8_t _val = bitRead(TasmotaGlobal.rel_inverted, i) ? !relay_state : relay_state;
-
- //AddLog(LOG_LEVEL_DEBUG, PSTR("PCF: SwitchRelay %d=%d => PCF-%d.D%d=%d"), i, relay_state, board +1, pin, _val);
- bitWrite(Pcf8574.pin_mask[board], pin, _val);
- if (oldpinmask != Pcf8574.pin_mask[board]) {
- Pcf8574.error = Pcf8574Write(board);
- }
- //else AddLog(LOG_LEVEL_DEBUG, PSTR("PCF: SwitchRelay skipped"));
- }
- }
-}
-
-void Pcf8574Init(void)
-{
- uint8_t pcf8574_address = (PCF8574_ADDR1_COUNT > 0) ? PCF8574_ADDR1 : PCF8574_ADDR2;
- while ((Pcf8574.max_devices < MAX_PCF8574) && (pcf8574_address < PCF8574_ADDR2 +PCF8574_ADDR2_COUNT)) {
-
-#if defined(USE_MCP230xx) && defined(USE_MCP230xx_ADDR)
- if (USE_MCP230xx_ADDR == pcf8574_address) {
- AddLog(LOG_LEVEL_INFO, PSTR("PCF: Address 0x%02x reserved for MCP320xx skipped"), pcf8574_address);
- pcf8574_address++;
- if ((PCF8574_ADDR1 +PCF8574_ADDR1_COUNT) == pcf8574_address) { // See comment on allowed addresses and overrides
- pcf8574_address = PCF8574_ADDR2;
- }
- }
-#endif
-
- // AddLog(LOG_LEVEL_DEBUG, PSTR("PCF: Probing addr: 0x%x for PCF8574"), pcf8574_address);
-
- if (I2cSetDevice(pcf8574_address)) {
- Pcf8574.type = true;
-
- Pcf8574.address[Pcf8574.max_devices] = pcf8574_address;
- Pcf8574.max_devices++;
-
- strcpy(Pcf8574.stype, "PCF8574");
- if (pcf8574_address >= PCF8574_ADDR2) {
- strcpy(Pcf8574.stype, "PCF8574A");
- }
- I2cSetActiveFound(pcf8574_address, Pcf8574.stype);
- }
-
- pcf8574_address++;
- if ((PCF8574_ADDR1 +PCF8574_ADDR1_COUNT) == pcf8574_address) { // Support I2C addresses 0x20 to 0x26 and 0x39 to 0x3F
- pcf8574_address = PCF8574_ADDR2;
- }
- }
- if (Pcf8574.type) {
- for (uint32_t i = 0; i < sizeof(Pcf8574.pin); i++) {
- Pcf8574.pin[i] = 99;
- }
- UpdateDevicesPresent(-Pcf8574.max_connected_ports); // reset no of devices to avoid duplicate ports on duplicate init.
-
- Pcf8574.max_connected_ports = 0; // reset no of devices to avoid duplicate ports on duplicate init.
- for (uint32_t idx = 0; idx < Pcf8574.max_devices; idx++) { // suport up to 8 boards PCF8574
- uint8_t gpio = Pcf8574Read(idx);
- // Insure the input pins are actually writen a 1 for proper input operation
- Pcf8574.pin_mask[idx] = gpio | ~Settings->pcf8574_config[idx];
- Pcf8574Write(idx); // Write back to the register
-#ifdef USE_PCF8574_MQTTINPUT
- Pcf8574.last_input[idx] = gpio & ~Settings->pcf8574_config[idx];
-#endif // #ifdef USE_PCF8574_MQTTINPUT
- //AddLog(LOG_LEVEL_DEBUG, PSTR("PCF: PCF-%d config=0x%02x, gpio=0x%02X"), idx +1, Settings->pcf8574_config[idx], gpio);
-
- for (uint32_t i = 0; i < 8; i++, gpio>>=1) {
- uint8_t _result = Settings->pcf8574_config[idx] >> i &1;
- //AddLog(LOG_LEVEL_DEBUG, PSTR("PCF: I2C shift i %d: %d. Powerstate: %d, TasmotaGlobal.devices_present: %d"), i,_result, Settings->power>>i&1, TasmotaGlobal.devices_present);
- if (_result > 0) {
- Pcf8574.pin[TasmotaGlobal.devices_present] = i + 8 * idx;
- bitWrite(TasmotaGlobal.rel_inverted, TasmotaGlobal.devices_present, Settings->flag3.pcf8574_ports_inverted); // SetOption81 - Invert all ports on PCF8574 devices
- if (!Settings->flag.save_state && !Settings->flag3.no_power_feedback) { // SetOption63 - Don't scan relay power state at restart - #5594 and #5663
- //AddLog(LOG_LEVEL_DEBUG, PSTR("PCF: Set power from from chip state"));
- uint8_t power_state = Settings->flag3.pcf8574_ports_inverted ? 1 & ~gpio : 1 & gpio;
- bitWrite(TasmotaGlobal.power, TasmotaGlobal.devices_present, power_state);
- bitWrite(Settings->power, TasmotaGlobal.devices_present, power_state);
- }
- //else AddLog(LOG_LEVEL_DEBUG, PSTR("PCF: DON'T set power from chip state"));
- UpdateDevicesPresent(1);
- Pcf8574.max_connected_ports++;
- }
- }
- }
- //AddLog(LOG_LEVEL_DEBUG, PSTR("PCF: Settings->power=0x%08X, TasmotaGlobal.power=0x%08X"), Settings->power, TasmotaGlobal.power);
- AddLog(LOG_LEVEL_INFO, PSTR("PCF: Total devices %d, PCF8574 output ports %d"), Pcf8574.max_devices, Pcf8574.max_connected_ports);
- }
-}
-
-/*********************************************************************************************\
- * Presentation
-\*********************************************************************************************/
-
-#ifdef USE_WEBSERVER
-
-#define WEB_HANDLE_PCF8574 "pcf"
-
-const char HTTP_BTN_MENU_PCF8574[] PROGMEM =
- "
";
-
-const char HTTP_FORM_I2C_PCF8574_1[] PROGMEM =
- "