From 024cc0b9b630be5aee8f8e92c8c19ebc816efde7 Mon Sep 17 00:00:00 2001 From: Marius Bezuidenhout Date: Mon, 12 Jul 2021 12:32:27 +0200 Subject: [PATCH 1/8] Added MCP2515 library and Freedom Won BMS sensor --- lib/lib_div/arduino-mcp2515-1.0.1/.project | 11 + .../org.eclipse.core.resources.prefs | 2 + lib/lib_div/arduino-mcp2515-1.0.1/LICENSE.md | 22 + lib/lib_div/arduino-mcp2515-1.0.1/README.md | 240 ++++++ lib/lib_div/arduino-mcp2515-1.0.1/can.h | 45 ++ .../examples/CAN_read/CAN_read.ino | 39 + .../examples/CAN_write/CAN_write.ino | 53 ++ .../arduino-mcp2515-1.0.1/keywords.txt | 14 + lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.cpp | 693 ++++++++++++++++++ lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h | 465 ++++++++++++ tasmota/i18n.h | 2 + tasmota/language/af_AF.h | 3 + tasmota/language/bg_BG.h | 1 + tasmota/language/cs_CZ.h | 3 + tasmota/language/de_DE.h | 3 + tasmota/language/el_GR.h | 3 + tasmota/language/en_GB.h | 3 + tasmota/language/es_ES.h | 3 + tasmota/language/fr_FR.h | 3 + tasmota/language/fy_NL.h | 3 + tasmota/language/he_HE.h | 3 + tasmota/language/hu_HU.h | 3 + tasmota/language/it_IT.h | 3 + tasmota/language/ko_KO.h | 3 + tasmota/language/nl_NL.h | 3 + tasmota/language/pl_PL.h | 3 + tasmota/language/pt_BR.h | 3 + tasmota/language/pt_PT.h | 3 + tasmota/language/ro_RO.h | 3 + tasmota/language/ru_RU.h | 3 + tasmota/language/sk_SK.h | 3 + tasmota/language/sv_SE.h | 3 + tasmota/language/tr_TR.h | 3 + tasmota/language/uk_UA.h | 3 + tasmota/language/vi_VN.h | 3 + tasmota/language/zh_CN.h | 3 + tasmota/language/zh_TW.h | 3 + tasmota/tasmota_template.h | 5 + tasmota/xsns_89_mcp2515.ino | 189 +++++ 39 files changed, 1856 insertions(+) create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/.project create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/.settings/org.eclipse.core.resources.prefs create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/LICENSE.md create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/README.md create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/can.h create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/examples/CAN_read/CAN_read.ino create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/examples/CAN_write/CAN_write.ino create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/keywords.txt create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.cpp create mode 100644 lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h create mode 100644 tasmota/xsns_89_mcp2515.ino diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/.project b/lib/lib_div/arduino-mcp2515-1.0.1/.project new file mode 100644 index 000000000000..0a2e228ff002 --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/.project @@ -0,0 +1,11 @@ + + + arduino-mcp2515 + + + + + + + + diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/.settings/org.eclipse.core.resources.prefs b/lib/lib_div/arduino-mcp2515-1.0.1/.settings/org.eclipse.core.resources.prefs new file mode 100644 index 000000000000..99f26c0203a7 --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/.settings/org.eclipse.core.resources.prefs @@ -0,0 +1,2 @@ +eclipse.preferences.version=1 +encoding/=UTF-8 diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/LICENSE.md b/lib/lib_div/arduino-mcp2515-1.0.1/LICENSE.md new file mode 100644 index 000000000000..168474e24745 --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/LICENSE.md @@ -0,0 +1,22 @@ +The MIT License (MIT) + +Copyright (c) 2013 Seeed Technology Inc. +Copyright (c) 2016 Dmitry + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/README.md b/lib/lib_div/arduino-mcp2515-1.0.1/README.md new file mode 100644 index 000000000000..f8fbbb25191d --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/README.md @@ -0,0 +1,240 @@ +Arduino MCP2515 CAN interface library +--------------------------------------------------------- + + +
+CAN-BUS is a common industrial bus because of its long travel distance, medium communication speed and high reliability. It is commonly found on modern machine tools and as an automotive diagnostic bus. This CAN-BUS Shield adopts MCP2515 CAN Bus controller with SPI interface and MCP2551 CAN transceiver to give your Arduino/Seeeduino CAN-BUS capibility. With an OBD-II converter cable added on and the OBD-II library imported, you are ready to build an onboard diagnostic device or data logger. + +- Implements CAN V2.0B at up to 1 Mb/s +- SPI Interface up to 10 MHz +- Standard (11 bit) and extended (29 bit) data and remote frames +- Two receive buffers with prioritized message storage + +## Library Installation + +1. Download the ZIP file from https://github.com/autowp/arduino-mcp2515/archive/master.zip +2. From the Arduino IDE: Sketch -> Include Library... -> Add .ZIP Library... +3. Restart the Arduino IDE to see the new "mcp2515" library with examples + + +
+# Usage: + + + +## 1. Initialization + +To create connection with MCP2515 provide pin number where SPI CS is connected (10 by default), baudrate and mode + +The available modes are listed as follows: +```C++ +mcp2515.setNormalMode(); +mcp2515.setLoopbackMode(); +mcp2515.setListenOnlyMode(); +``` +The available baudrates are listed as follows: +```C++ +enum CAN_SPEED { + CAN_5KBPS, + CAN_10KBPS, + CAN_20KBPS, + CAN_31K25BPS, + CAN_33KBPS, + CAN_40KBPS, + CAN_50KBPS, + CAN_80KBPS, + CAN_83K3BPS, + CAN_95KBPS, + CAN_100KBPS, + CAN_125KBPS, + CAN_200KBPS, + CAN_250KBPS, + CAN_500KBPS, + CAN_1000KBPS +}; +``` + +
+Example of initialization +```C++ +MCP2515 mcp2515(10); +mcp2515.reset(); +mcp2515.setBitrate(CAN_125KBPS); +mcp2515.setLoopbackMode(); +``` +
+ +
+You can also set oscillator frequency for module when setting bitrate: +```C++ +mcp2515.setBitrate(CAN_125KBPS, MCP_8MHZ); +``` +
+The available clock speeds are listed as follows: +```C++ +enum CAN_CLOCK { + MCP_20MHZ, + MCP_16MHZ, + MCP_8MHZ +}; +``` +Default value is MCP_16MHZ +
+ +Note: To transfer data on high speed of CAN interface via UART dont forget to update UART baudrate as necessary. + +##2. Frame data format + +Library uses Linux-like structure to store can frames; +```C++ +struct can_frame { + uint32_t can_id; /* 32 bit CAN_ID + EFF/RTR/ERR flags */ + uint8_t can_dlc; + uint8_t data[8]; +}; +``` +For additional information see [SocketCAN](https://www.kernel.org/doc/Documentation/networking/can.txt) + +## 3. Send Data +```C++ +MCP2515::ERROR sendMessage(const MCP2515::TXBn txbn, const struct can_frame *frame); +MCP2515::ERROR sendMessage(const struct can_frame *frame); +``` + +This is a function to send data onto the bus. + +For example, In the 'send' example, we have: + +```C++ +struct can_frame frame; +frame.can_id = 0x000; +frame.can_dlc = 4; +frame.data[0] = 0xFF; +frame.data[1] = 0xFF; +frame.data[2] = 0xFF; +frame.data[3] = 0xFF; + +/* send out the message to the bus and +tell other devices this is a standard frame from 0x00. */ +mcp2515.sendMessage(&frame); +``` + +```C++ +struct can_frame frame; +frame.can_id = 0x12345678 | CAN_EFF_MASK; +frame.can_dlc = 2; +frame.data[0] = 0xFF; +frame.data[1] = 0xFF; + +/* send out the message to the bus using second TX buffer and +tell other devices this is a extended frame from 0x12345678. */ +mcp2515.sendMessage(MCP2515::TXB1, &frame); +``` + + +
+## 4. Receive Data + +The following function is used to receive data on the 'receive' node: + +```C++ +MCP2515::ERROR readMessage(const MCP2515::RXBn rxbn, struct can_frame *frame); +MCP2515::ERROR readMessage(struct can_frame *frame); +``` + +In conditions that masks and filters have been set. This function can only get frames that meet the requirements of masks and filters. + +You can choise one of two method to receive: interrup-based and polling + +Example of poll read + +```C++ +struct can_frame frame; + +void loop() { + if (mcp2515.readMessage(&frame) == MCP2515::ERROR_OK) { + // frame contains received message + } +} +``` + +Example of interrupt based read +```C++ +bool interrupt = false; +struct can_frame frame; + +void irqHandler() { + interrupt = true; +} + +void setup() { + ... + attachInterrupt(0, irqHandler, FALLING); +} + +void loop() { + if (interrupt) { + interrupt = false; + + uint8_t irq = mcp2515.getInterrupts(); + + if (irq & MCP2515::CANINTF_RX0IF) { + if (mcp2515.readMessage(MCP2515::RXB0, &frame) == MCP2515::ERROR_OK) { + // frame contains received from RXB0 message + } + } + + if (irq & MCP2515::CANINTF_RX1IF) { + if (mcp2515.readMessage(MCP2515::RXB1, &frame) == MCP2515::ERROR_OK) { + // frame contains received from RXB1 message + } + } + } +} +``` +
+##5. Set Receive Mask and Filter + +There are 2 receive mask registers and 5 filter registers on the controller chip that guarantee you get data from the target device. They are useful especially in a large network consisting of numerous nodes. + +We provide two functions for you to utilize these mask and filter registers. They are: + +```C++ +MCP2515::ERROR setFilterMask(const MASK mask, const bool ext, const uint32_t ulData) +MCP2515::ERROR setFilter(const RXF num, const bool ext, const uint32_t ulData) +``` + +**MASK mask** represents one of two mask **MCP2515::MASK0** or **MCP2515::MASK1** + +**RXF num** represents one of six acceptance filters registers from **MCP2515::RXF0** to **MCP2515::RXF5** + +**ext** represents the status of the frame. **false** means it's a mask or filter for a standard frame. **true** means it's for a extended frame. + +**ulData** represents the content of the mask of filter. + + + +
+## 6. Examples + +Example implementation of CanHacker (lawicel) protocol based device: [https://github.com/autowp/can-usb](https://github.com/autowp/can-usb) + +
+For more information, please refer to [wiki page](http://www.seeedstudio.com/wiki/CAN-BUS_Shield). + + +---- + +This software is written by loovee ([luweicong@seeed.cc](luweicong@seeed.cc "luweicong@seeed.cc")) for seeed studio,
+Updated by Dmitry ([https://github.com/autowp](https://github.com/autowp "https://github.com/autowp"))
+and is licensed under [The MIT License](http://opensource.org/licenses/mit-license.php). Check [LICENSE.md](LICENSE.md) for more information.
+ +Contributing to this software is warmly welcomed. You can do this basically by
+[forking](https://help.github.com/articles/fork-a-repo), committing modifications and then [pulling requests](https://help.github.com/articles/using-pull-requests) (follow the links above
+for operating guide). Adding change log and your contact into file header is encouraged.
+Thanks for your contribution. + +Seeed Studio is an open hardware facilitation company based in Shenzhen, China.
+Benefiting from local manufacture power and convenient global logistic system,
+we integrate resources to serve new era of innovation. Seeed also works with
+global distributors and partners to push open hardware movement.
diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/can.h b/lib/lib_div/arduino-mcp2515-1.0.1/can.h new file mode 100644 index 000000000000..96175b1f2265 --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/can.h @@ -0,0 +1,45 @@ +#ifndef CAN_H_ +#define CAN_H_ + +#include + + +typedef unsigned char __u8; +typedef unsigned short __u16; +typedef unsigned long __u32; + + +/* special address description flags for the CAN_ID */ +#define CAN_EFF_FLAG 0x80000000UL /* EFF/SFF is set in the MSB */ +#define CAN_RTR_FLAG 0x40000000UL /* remote transmission request */ +#define CAN_ERR_FLAG 0x20000000UL /* error message frame */ + +/* valid bits in CAN ID for frame formats */ +#define CAN_SFF_MASK 0x000007FFUL /* standard frame format (SFF) */ +#define CAN_EFF_MASK 0x1FFFFFFFUL /* extended frame format (EFF) */ +#define CAN_ERR_MASK 0x1FFFFFFFUL /* omit EFF, RTR, ERR flags */ + +/* + * Controller Area Network Identifier structure + * + * bit 0-28 : CAN identifier (11/29 bit) + * bit 29 : error message frame flag (0 = data frame, 1 = error message) + * bit 30 : remote transmission request flag (1 = rtr frame) + * bit 31 : frame format flag (0 = standard 11 bit, 1 = extended 29 bit) + */ +typedef __u32 canid_t; + +#define CAN_SFF_ID_BITS 11 +#define CAN_EFF_ID_BITS 29 + +/* CAN payload length and DLC definitions according to ISO 11898-1 */ +#define CAN_MAX_DLC 8 +#define CAN_MAX_DLEN 8 + +struct can_frame { + canid_t can_id; /* 32 bit CAN_ID + EFF/RTR/ERR flags */ + __u8 can_dlc; /* frame payload length in byte (0 .. CAN_MAX_DLEN) */ + __u8 data[CAN_MAX_DLEN] __attribute__((aligned(8))); +}; + +#endif /* CAN_H_ */ diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/examples/CAN_read/CAN_read.ino b/lib/lib_div/arduino-mcp2515-1.0.1/examples/CAN_read/CAN_read.ino new file mode 100644 index 000000000000..b43e56a52af6 --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/examples/CAN_read/CAN_read.ino @@ -0,0 +1,39 @@ +#include +#include + +struct can_frame canMsg; +MCP2515 mcp2515(10); + + +void setup() { + Serial.begin(115200); + SPI.begin(); + + mcp2515.reset(); + mcp2515.setBitrate(CAN_125KBPS); + mcp2515.setNormalMode(); + + Serial.println("------- CAN Read ----------"); + Serial.println("ID DLC DATA"); +} + +void loop() { + + if (mcp2515.readMessage(&canMsg) == MCP2515::ERROR_OK) { + + Serial.print(canMsg.can_id, HEX); // print ID + Serial.print(" "); + Serial.print(canMsg.can_dlc, HEX); // print DLC + Serial.print(" "); + + for (int i = 0; i +#include + +struct can_frame canMsg1; +struct can_frame canMsg2; +MCP2515 mcp2515(10); + + +void setup() { + + canMsg1.can_id = 0x0F6; + canMsg1.can_dlc = 8; + canMsg1.data[0] = 0x8E; + canMsg1.data[1] = 0x87; + canMsg1.data[2] = 0x32; + canMsg1.data[3] = 0xFA; + canMsg1.data[4] = 0x26; + canMsg1.data[5] = 0x8E; + canMsg1.data[6] = 0xBE; + canMsg1.data[7] = 0x86; + + canMsg2.can_id = 0x036; + canMsg2.can_dlc = 8; + canMsg2.data[0] = 0x0E; + canMsg2.data[1] = 0x00; + canMsg2.data[2] = 0x00; + canMsg2.data[3] = 0x08; + canMsg2.data[4] = 0x01; + canMsg2.data[5] = 0x00; + canMsg2.data[6] = 0x00; + canMsg2.data[7] = 0xA0; + + while (!Serial); + Serial.begin(115200); + SPI.begin(); + + mcp2515.reset(); + mcp2515.setBitrate(CAN_125KBPS); + mcp2515.setNormalMode(); + + Serial.println("Example: Write to CAN"); +} + +void loop() { + + mcp2515.sendMessage(&canMsg1); + mcp2515.sendMessage(&canMsg2); + + Serial.println("Messages sent"); + + delay(100); + +} diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/keywords.txt b/lib/lib_div/arduino-mcp2515-1.0.1/keywords.txt new file mode 100644 index 000000000000..9dd85de32f1f --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/keywords.txt @@ -0,0 +1,14 @@ +MCP2515 KEYWORD1 + +CAN_5KBPS LITERAL1 +CAN_10KBPS LITERAL1 +CAN_20KBPS LITERAL1 +CAN_40KBPS LITERAL1 +CAN_50KBPS LITERAL1 +CAN_80KBPS LITERAL1 +CAN_100KBPS LITERAL1 +CAN_125KBPS LITERAL1 +CAN_200KBPS LITERAL1 +CAN_250KBPS LITERAL1 +CAN_500KBPS LITERAL1 +CAN_1000KBPS LITERAL1 diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.cpp b/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.cpp new file mode 100644 index 000000000000..efa7c795c658 --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.cpp @@ -0,0 +1,693 @@ +#include "mcp2515.h" + +const struct MCP2515::TXBn_REGS MCP2515::TXB[MCP2515::N_TXBUFFERS] = { + {MCP_TXB0CTRL, MCP_TXB0SIDH, MCP_TXB0DATA}, + {MCP_TXB1CTRL, MCP_TXB1SIDH, MCP_TXB1DATA}, + {MCP_TXB2CTRL, MCP_TXB2SIDH, MCP_TXB2DATA} +}; + +const struct MCP2515::RXBn_REGS MCP2515::RXB[N_RXBUFFERS] = { + {MCP_RXB0CTRL, MCP_RXB0SIDH, MCP_RXB0DATA, CANINTF_RX0IF}, + {MCP_RXB1CTRL, MCP_RXB1SIDH, MCP_RXB1DATA, CANINTF_RX1IF} +}; + +MCP2515::MCP2515(const uint8_t _CS) +{ + SPI.begin(); + + SPICS = _CS; + pinMode(SPICS, OUTPUT); + endSPI(); +} + +void MCP2515::startSPI() { + SPI.beginTransaction(SPISettings(SPI_CLOCK, MSBFIRST, SPI_MODE0)); + digitalWrite(SPICS, LOW); +} + +void MCP2515::endSPI() { + digitalWrite(SPICS, HIGH); + SPI.endTransaction(); +} + +MCP2515::ERROR MCP2515::reset(void) +{ + startSPI(); + SPI.transfer(INSTRUCTION_RESET); + endSPI(); + + delay(10); + + uint8_t zeros[14]; + memset(zeros, 0, sizeof(zeros)); + setRegisters(MCP_TXB0CTRL, zeros, 14); + setRegisters(MCP_TXB1CTRL, zeros, 14); + setRegisters(MCP_TXB2CTRL, zeros, 14); + + setRegister(MCP_RXB0CTRL, 0); + setRegister(MCP_RXB1CTRL, 0); + + setRegister(MCP_CANINTE, CANINTF_RX0IF | CANINTF_RX1IF | CANINTF_ERRIF | CANINTF_MERRF); + + modifyRegister(MCP_RXB0CTRL, + RXBnCTRL_RXM_MASK | RXB0CTRL_BUKT, + RXBnCTRL_RXM_STDEXT | RXB0CTRL_BUKT); + modifyRegister(MCP_RXB1CTRL, RXBnCTRL_RXM_MASK, RXBnCTRL_RXM_STDEXT); + + // clear filters and masks + /*RXF filters[] = {RXF0, RXF1, RXF2, RXF3, RXF4, RXF5}; + for (int i=0; i<6; i++) { + ERROR result = setFilter(filters[i], true, 0); + if (result != ERROR_OK) { + return result; + } + } + + MASK masks[] = {MASK0, MASK1}; + for (int i=0; i<2; i++) { + ERROR result = setFilterMask(masks[i], true, 0); + if (result != ERROR_OK) { + return result; + } + }*/ + + return ERROR_OK; +} + +uint8_t MCP2515::readRegister(const REGISTER reg) +{ + startSPI(); + SPI.transfer(INSTRUCTION_READ); + SPI.transfer(reg); + uint8_t ret = SPI.transfer(0x00); + endSPI(); + + return ret; +} + +void MCP2515::readRegisters(const REGISTER reg, uint8_t values[], const uint8_t n) +{ + startSPI(); + SPI.transfer(INSTRUCTION_READ); + SPI.transfer(reg); + // mcp2515 has auto-increment of address-pointer + for (uint8_t i=0; i> 8); + canid = (uint16_t)(id >> 16); + buffer[MCP_SIDL] = (uint8_t) (canid & 0x03); + buffer[MCP_SIDL] += (uint8_t) ((canid & 0x1C) << 3); + buffer[MCP_SIDL] |= TXB_EXIDE_MASK; + buffer[MCP_SIDH] = (uint8_t) (canid >> 5); + } else { + buffer[MCP_SIDH] = (uint8_t) (canid >> 3); + buffer[MCP_SIDL] = (uint8_t) ((canid & 0x07 ) << 5); + buffer[MCP_EID0] = 0; + buffer[MCP_EID8] = 0; + } +} + +MCP2515::ERROR MCP2515::setFilterMask(const MASK mask, const bool ext, const uint32_t ulData) +{ + ERROR res = setConfigMode(); + if (res != ERROR_OK) { + return res; + } + + uint8_t tbufdata[4]; + prepareId(tbufdata, ext, ulData); + + REGISTER reg; + switch (mask) { + case MASK0: reg = MCP_RXM0SIDH; break; + case MASK1: reg = MCP_RXM1SIDH; break; + default: + return ERROR_FAIL; + } + + setRegisters(reg, tbufdata, 4); + + return ERROR_OK; +} + +MCP2515::ERROR MCP2515::setFilter(const RXF num, const bool ext, const uint32_t ulData) +{ + ERROR res = setConfigMode(); + if (res != ERROR_OK) { + return res; + } + + REGISTER reg; + + switch (num) { + case RXF0: reg = MCP_RXF0SIDH; break; + case RXF1: reg = MCP_RXF1SIDH; break; + case RXF2: reg = MCP_RXF2SIDH; break; + case RXF3: reg = MCP_RXF3SIDH; break; + case RXF4: reg = MCP_RXF4SIDH; break; + case RXF5: reg = MCP_RXF5SIDH; break; + default: + return ERROR_FAIL; + } + + uint8_t tbufdata[4]; + prepareId(tbufdata, ext, ulData); + setRegisters(reg, tbufdata, 4); + + return ERROR_OK; +} + +MCP2515::ERROR MCP2515::sendMessage(const TXBn txbn, const struct can_frame *frame) +{ + const struct TXBn_REGS *txbuf = &TXB[txbn]; + + uint8_t data[13]; + + bool ext = (frame->can_id & CAN_EFF_FLAG); + bool rtr = (frame->can_id & CAN_RTR_FLAG); + uint32_t id = (frame->can_id & (ext ? CAN_EFF_MASK : CAN_SFF_MASK)); + + prepareId(data, ext, id); + + data[MCP_DLC] = rtr ? (frame->can_dlc | RTR_MASK) : frame->can_dlc; + + memcpy(&data[MCP_DATA], frame->data, frame->can_dlc); + + setRegisters(txbuf->SIDH, data, 5 + frame->can_dlc); + + modifyRegister(txbuf->CTRL, TXB_TXREQ, TXB_TXREQ); + + return ERROR_OK; +} + +MCP2515::ERROR MCP2515::sendMessage(const struct can_frame *frame) +{ + if (frame->can_dlc > CAN_MAX_DLEN) { + return ERROR_FAILTX; + } + + TXBn txBuffers[N_TXBUFFERS] = {TXB0, TXB1, TXB2}; + + for (int i=0; iCTRL); + if ( (ctrlval & TXB_TXREQ) == 0 ) { + return sendMessage(txBuffers[i], frame); + } + } + + return ERROR_FAILTX; +} + +MCP2515::ERROR MCP2515::readMessage(const RXBn rxbn, struct can_frame *frame) +{ + const struct RXBn_REGS *rxb = &RXB[rxbn]; + + uint8_t tbufdata[5]; + + readRegisters(rxb->SIDH, tbufdata, 5); + + uint32_t id = (tbufdata[MCP_SIDH]<<3) + (tbufdata[MCP_SIDL]>>5); + + if ( (tbufdata[MCP_SIDL] & TXB_EXIDE_MASK) == TXB_EXIDE_MASK ) { + id = (id<<2) + (tbufdata[MCP_SIDL] & 0x03); + id = (id<<8) + tbufdata[MCP_EID8]; + id = (id<<8) + tbufdata[MCP_EID0]; + id |= CAN_EFF_FLAG; + } + + uint8_t dlc = (tbufdata[MCP_DLC] & DLC_MASK); + if (dlc > CAN_MAX_DLEN) { + return ERROR_FAIL; + } + + uint8_t ctrl = readRegister(rxb->CTRL); + if (ctrl & RXBnCTRL_RTR) { + id |= CAN_RTR_FLAG; + } + + frame->can_id = id; + frame->can_dlc = dlc; + + readRegisters(rxb->DATA, frame->data, dlc); + + modifyRegister(MCP_CANINTF, rxb->CANINTF_RXnIF, 0); + + return ERROR_OK; +} + +MCP2515::ERROR MCP2515::readMessage(struct can_frame *frame) +{ + ERROR rc; + uint8_t stat = getStatus(); + + if ( stat & STAT_RX0IF ) { + rc = readMessage(RXB0, frame); + } else if ( stat & STAT_RX1IF ) { + rc = readMessage(RXB1, frame); + } else { + rc = ERROR_NOMSG; + } + + return rc; +} + +bool MCP2515::checkReceive(void) +{ + uint8_t res = getStatus(); + if ( res & STAT_RXIF_MASK ) { + return true; + } else { + return false; + } +} + +bool MCP2515::checkError(void) +{ + uint8_t eflg = getErrorFlags(); + + if ( eflg & EFLG_ERRORMASK ) { + return true; + } else { + return false; + } +} + +uint8_t MCP2515::getErrorFlags(void) +{ + return readRegister(MCP_EFLG); +} + +void MCP2515::clearRXnOVRFlags(void) +{ + modifyRegister(MCP_EFLG, EFLG_RX0OVR | EFLG_RX1OVR, 0); +} + +uint8_t MCP2515::getInterrupts(void) +{ + return readRegister(MCP_CANINTF); +} + +void MCP2515::clearInterrupts(void) +{ + setRegister(MCP_CANINTF, 0); +} + +uint8_t MCP2515::getInterruptMask(void) +{ + return readRegister(MCP_CANINTE); +} + +void MCP2515::clearTXInterrupts(void) +{ + modifyRegister(MCP_CANINTF, (CANINTF_TX0IF | CANINTF_TX1IF | CANINTF_TX2IF), 0); +} + +void MCP2515::clearRXnOVR(void) +{ + uint8_t eflg = getErrorFlags(); + if (eflg != 0) { + clearRXnOVRFlags(); + clearInterrupts(); + //modifyRegister(MCP_CANINTF, CANINTF_ERRIF, 0); + } + +} + +void MCP2515::clearMERR() +{ + //modifyRegister(MCP_EFLG, EFLG_RX0OVR | EFLG_RX1OVR, 0); + //clearInterrupts(); + modifyRegister(MCP_CANINTF, CANINTF_MERRF, 0); +} diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h b/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h new file mode 100644 index 000000000000..a73d1500e578 --- /dev/null +++ b/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h @@ -0,0 +1,465 @@ +#ifndef _MCP2515_H_ +#define _MCP2515_H_ + +#include +#include "can.h" + +/* + * Speed 8M + */ +#define MCP_8MHz_1000kBPS_CFG1 (0x00) +#define MCP_8MHz_1000kBPS_CFG2 (0x80) +#define MCP_8MHz_1000kBPS_CFG3 (0x80) + +#define MCP_8MHz_500kBPS_CFG1 (0x00) +#define MCP_8MHz_500kBPS_CFG2 (0x90) +#define MCP_8MHz_500kBPS_CFG3 (0x82) + +#define MCP_8MHz_250kBPS_CFG1 (0x00) +#define MCP_8MHz_250kBPS_CFG2 (0xB1) +#define MCP_8MHz_250kBPS_CFG3 (0x85) + +#define MCP_8MHz_200kBPS_CFG1 (0x00) +#define MCP_8MHz_200kBPS_CFG2 (0xB4) +#define MCP_8MHz_200kBPS_CFG3 (0x86) + +#define MCP_8MHz_125kBPS_CFG1 (0x01) +#define MCP_8MHz_125kBPS_CFG2 (0xB1) +#define MCP_8MHz_125kBPS_CFG3 (0x85) + +#define MCP_8MHz_100kBPS_CFG1 (0x01) +#define MCP_8MHz_100kBPS_CFG2 (0xB4) +#define MCP_8MHz_100kBPS_CFG3 (0x86) + +#define MCP_8MHz_80kBPS_CFG1 (0x01) +#define MCP_8MHz_80kBPS_CFG2 (0xBF) +#define MCP_8MHz_80kBPS_CFG3 (0x87) + +#define MCP_8MHz_50kBPS_CFG1 (0x03) +#define MCP_8MHz_50kBPS_CFG2 (0xB4) +#define MCP_8MHz_50kBPS_CFG3 (0x86) + +#define MCP_8MHz_40kBPS_CFG1 (0x03) +#define MCP_8MHz_40kBPS_CFG2 (0xBF) +#define MCP_8MHz_40kBPS_CFG3 (0x87) + +#define MCP_8MHz_33k3BPS_CFG1 (0x47) +#define MCP_8MHz_33k3BPS_CFG2 (0xE2) +#define MCP_8MHz_33k3BPS_CFG3 (0x85) + +#define MCP_8MHz_31k25BPS_CFG1 (0x07) +#define MCP_8MHz_31k25BPS_CFG2 (0xA4) +#define MCP_8MHz_31k25BPS_CFG3 (0x84) + +#define MCP_8MHz_20kBPS_CFG1 (0x07) +#define MCP_8MHz_20kBPS_CFG2 (0xBF) +#define MCP_8MHz_20kBPS_CFG3 (0x87) + +#define MCP_8MHz_10kBPS_CFG1 (0x0F) +#define MCP_8MHz_10kBPS_CFG2 (0xBF) +#define MCP_8MHz_10kBPS_CFG3 (0x87) + +#define MCP_8MHz_5kBPS_CFG1 (0x1F) +#define MCP_8MHz_5kBPS_CFG2 (0xBF) +#define MCP_8MHz_5kBPS_CFG3 (0x87) + +/* + * speed 16M + */ +#define MCP_16MHz_1000kBPS_CFG1 (0x00) +#define MCP_16MHz_1000kBPS_CFG2 (0xD0) +#define MCP_16MHz_1000kBPS_CFG3 (0x82) + +#define MCP_16MHz_500kBPS_CFG1 (0x00) +#define MCP_16MHz_500kBPS_CFG2 (0xF0) +#define MCP_16MHz_500kBPS_CFG3 (0x86) + +#define MCP_16MHz_250kBPS_CFG1 (0x41) +#define MCP_16MHz_250kBPS_CFG2 (0xF1) +#define MCP_16MHz_250kBPS_CFG3 (0x85) + +#define MCP_16MHz_200kBPS_CFG1 (0x01) +#define MCP_16MHz_200kBPS_CFG2 (0xFA) +#define MCP_16MHz_200kBPS_CFG3 (0x87) + +#define MCP_16MHz_125kBPS_CFG1 (0x03) +#define MCP_16MHz_125kBPS_CFG2 (0xF0) +#define MCP_16MHz_125kBPS_CFG3 (0x86) + +#define MCP_16MHz_100kBPS_CFG1 (0x03) +#define MCP_16MHz_100kBPS_CFG2 (0xFA) +#define MCP_16MHz_100kBPS_CFG3 (0x87) + +#define MCP_16MHz_80kBPS_CFG1 (0x03) +#define MCP_16MHz_80kBPS_CFG2 (0xFF) +#define MCP_16MHz_80kBPS_CFG3 (0x87) + +#define MCP_16MHz_50kBPS_CFG1 (0x07) +#define MCP_16MHz_50kBPS_CFG2 (0xFA) +#define MCP_16MHz_50kBPS_CFG3 (0x87) + +#define MCP_16MHz_40kBPS_CFG1 (0x07) +#define MCP_16MHz_40kBPS_CFG2 (0xFF) +#define MCP_16MHz_40kBPS_CFG3 (0x87) + +#define MCP_16MHz_33k3BPS_CFG1 (0x4E) +#define MCP_16MHz_33k3BPS_CFG2 (0xF1) +#define MCP_16MHz_33k3BPS_CFG3 (0x85) + +#define MCP_16MHz_20kBPS_CFG1 (0x0F) +#define MCP_16MHz_20kBPS_CFG2 (0xFF) +#define MCP_16MHz_20kBPS_CFG3 (0x87) + +#define MCP_16MHz_10kBPS_CFG1 (0x1F) +#define MCP_16MHz_10kBPS_CFG2 (0xFF) +#define MCP_16MHz_10kBPS_CFG3 (0x87) + +#define MCP_16MHz_5kBPS_CFG1 (0x3F) +#define MCP_16MHz_5kBPS_CFG2 (0xFF) +#define MCP_16MHz_5kBPS_CFG3 (0x87) + +/* + * speed 20M + */ +#define MCP_20MHz_1000kBPS_CFG1 (0x00) +#define MCP_20MHz_1000kBPS_CFG2 (0xD9) +#define MCP_20MHz_1000kBPS_CFG3 (0x82) + +#define MCP_20MHz_500kBPS_CFG1 (0x00) +#define MCP_20MHz_500kBPS_CFG2 (0xFA) +#define MCP_20MHz_500kBPS_CFG3 (0x87) + +#define MCP_20MHz_250kBPS_CFG1 (0x41) +#define MCP_20MHz_250kBPS_CFG2 (0xFB) +#define MCP_20MHz_250kBPS_CFG3 (0x86) + +#define MCP_20MHz_200kBPS_CFG1 (0x01) +#define MCP_20MHz_200kBPS_CFG2 (0xFF) +#define MCP_20MHz_200kBPS_CFG3 (0x87) + +#define MCP_20MHz_125kBPS_CFG1 (0x03) +#define MCP_20MHz_125kBPS_CFG2 (0xFA) +#define MCP_20MHz_125kBPS_CFG3 (0x87) + +#define MCP_20MHz_100kBPS_CFG1 (0x04) +#define MCP_20MHz_100kBPS_CFG2 (0xFA) +#define MCP_20MHz_100kBPS_CFG3 (0x87) + +#define MCP_20MHz_80kBPS_CFG1 (0x04) +#define MCP_20MHz_80kBPS_CFG2 (0xFF) +#define MCP_20MHz_80kBPS_CFG3 (0x87) + +#define MCP_20MHz_50kBPS_CFG1 (0x09) +#define MCP_20MHz_50kBPS_CFG2 (0xFA) +#define MCP_20MHz_50kBPS_CFG3 (0x87) + +#define MCP_20MHz_40kBPS_CFG1 (0x09) +#define MCP_20MHz_40kBPS_CFG2 (0xFF) +#define MCP_20MHz_40kBPS_CFG3 (0x87) + +enum CAN_CLOCK { + MCP_20MHZ, + MCP_16MHZ, + MCP_8MHZ +}; + +enum CAN_SPEED { + CAN_5KBPS, + CAN_10KBPS, + CAN_20KBPS, + CAN_31K25BPS, + CAN_33KBPS, + CAN_40KBPS, + CAN_50KBPS, + CAN_80KBPS, + CAN_83K3BPS, + CAN_95KBPS, + CAN_100KBPS, + CAN_125KBPS, + CAN_200KBPS, + CAN_250KBPS, + CAN_500KBPS, + CAN_1000KBPS +}; + +class MCP2515 +{ + public: + enum ERROR { + ERROR_OK = 0, + ERROR_FAIL = 1, + ERROR_ALLTXBUSY = 2, + ERROR_FAILINIT = 3, + ERROR_FAILTX = 4, + ERROR_NOMSG = 5 + }; + + enum MASK { + MASK0, + MASK1 + }; + + enum RXF { + RXF0 = 0, + RXF1 = 1, + RXF2 = 2, + RXF3 = 3, + RXF4 = 4, + RXF5 = 5 + }; + + enum RXBn { + RXB0 = 0, + RXB1 = 1 + }; + + enum TXBn { + TXB0 = 0, + TXB1 = 1, + TXB2 = 2 + }; + + enum /*class*/ CANINTF : uint8_t { + CANINTF_RX0IF = 0x01, + CANINTF_RX1IF = 0x02, + CANINTF_TX0IF = 0x04, + CANINTF_TX1IF = 0x08, + CANINTF_TX2IF = 0x10, + CANINTF_ERRIF = 0x20, + CANINTF_WAKIF = 0x40, + CANINTF_MERRF = 0x80 + }; + + private: + static const uint8_t CANCTRL_REQOP = 0xE0; + static const uint8_t CANCTRL_ABAT = 0x10; + static const uint8_t CANCTRL_OSM = 0x08; + static const uint8_t CANCTRL_CLKEN = 0x04; + static const uint8_t CANCTRL_CLKPRE = 0x03; + + enum /*class*/ CANCTRL_REQOP_MODE : uint8_t { + CANCTRL_REQOP_NORMAL = 0x00, + CANCTRL_REQOP_SLEEP = 0x20, + CANCTRL_REQOP_LOOPBACK = 0x40, + CANCTRL_REQOP_LISTENONLY = 0x60, + CANCTRL_REQOP_CONFIG = 0x80, + CANCTRL_REQOP_POWERUP = 0xE0 + }; + + static const uint8_t CANSTAT_OPMOD = 0xE0; + static const uint8_t CANSTAT_ICOD = 0x0E; + + static const uint8_t TXB_EXIDE_MASK = 0x08; + static const uint8_t DLC_MASK = 0x0F; + static const uint8_t RTR_MASK = 0x40; + + static const uint8_t RXBnCTRL_RXM_STD = 0x20; + static const uint8_t RXBnCTRL_RXM_EXT = 0x40; + static const uint8_t RXBnCTRL_RXM_STDEXT = 0x00; + static const uint8_t RXBnCTRL_RXM_MASK = 0x60; + static const uint8_t RXBnCTRL_RTR = 0x08; + static const uint8_t RXB0CTRL_BUKT = 0x04; + + static const uint8_t MCP_SIDH = 0; + static const uint8_t MCP_SIDL = 1; + static const uint8_t MCP_EID8 = 2; + static const uint8_t MCP_EID0 = 3; + static const uint8_t MCP_DLC = 4; + static const uint8_t MCP_DATA = 5; + + enum /*class*/ STAT : uint8_t { + STAT_RX0IF = (1<<0), + STAT_RX1IF = (1<<1) + }; + + static const uint8_t STAT_RXIF_MASK = STAT_RX0IF | STAT_RX1IF; + + enum /*class*/ TXBnCTRL : uint8_t { + TXB_ABTF = 0x40, + TXB_MLOA = 0x20, + TXB_TXERR = 0x10, + TXB_TXREQ = 0x08, + TXB_TXIE = 0x04, + TXB_TXP = 0x03 + }; + + enum /*class*/ EFLG : uint8_t { + EFLG_RX1OVR = (1<<7), + EFLG_RX0OVR = (1<<6), + EFLG_TXBO = (1<<5), + EFLG_TXEP = (1<<4), + EFLG_RXEP = (1<<3), + EFLG_TXWAR = (1<<2), + EFLG_RXWAR = (1<<1), + EFLG_EWARN = (1<<0) + }; + + static const uint8_t EFLG_ERRORMASK = EFLG_RX1OVR + | EFLG_RX0OVR + | EFLG_TXBO + | EFLG_TXEP + | EFLG_RXEP; + + enum /*class*/ INSTRUCTION : uint8_t { + INSTRUCTION_WRITE = 0x02, + INSTRUCTION_READ = 0x03, + INSTRUCTION_BITMOD = 0x05, + INSTRUCTION_LOAD_TX0 = 0x40, + INSTRUCTION_LOAD_TX1 = 0x42, + INSTRUCTION_LOAD_TX2 = 0x44, + INSTRUCTION_RTS_TX0 = 0x81, + INSTRUCTION_RTS_TX1 = 0x82, + INSTRUCTION_RTS_TX2 = 0x84, + INSTRUCTION_RTS_ALL = 0x87, + INSTRUCTION_READ_RX0 = 0x90, + INSTRUCTION_READ_RX1 = 0x94, + INSTRUCTION_READ_STATUS = 0xA0, + INSTRUCTION_RX_STATUS = 0xB0, + INSTRUCTION_RESET = 0xC0 + }; + + enum /*class*/ REGISTER : uint8_t { + MCP_RXF0SIDH = 0x00, + MCP_RXF0SIDL = 0x01, + MCP_RXF0EID8 = 0x02, + MCP_RXF0EID0 = 0x03, + MCP_RXF1SIDH = 0x04, + MCP_RXF1SIDL = 0x05, + MCP_RXF1EID8 = 0x06, + MCP_RXF1EID0 = 0x07, + MCP_RXF2SIDH = 0x08, + MCP_RXF2SIDL = 0x09, + MCP_RXF2EID8 = 0x0A, + MCP_RXF2EID0 = 0x0B, + MCP_CANSTAT = 0x0E, + MCP_CANCTRL = 0x0F, + MCP_RXF3SIDH = 0x10, + MCP_RXF3SIDL = 0x11, + MCP_RXF3EID8 = 0x12, + MCP_RXF3EID0 = 0x13, + MCP_RXF4SIDH = 0x14, + MCP_RXF4SIDL = 0x15, + MCP_RXF4EID8 = 0x16, + MCP_RXF4EID0 = 0x17, + MCP_RXF5SIDH = 0x18, + MCP_RXF5SIDL = 0x19, + MCP_RXF5EID8 = 0x1A, + MCP_RXF5EID0 = 0x1B, + MCP_TEC = 0x1C, + MCP_REC = 0x1D, + MCP_RXM0SIDH = 0x20, + MCP_RXM0SIDL = 0x21, + MCP_RXM0EID8 = 0x22, + MCP_RXM0EID0 = 0x23, + MCP_RXM1SIDH = 0x24, + MCP_RXM1SIDL = 0x25, + MCP_RXM1EID8 = 0x26, + MCP_RXM1EID0 = 0x27, + MCP_CNF3 = 0x28, + MCP_CNF2 = 0x29, + MCP_CNF1 = 0x2A, + MCP_CANINTE = 0x2B, + MCP_CANINTF = 0x2C, + MCP_EFLG = 0x2D, + MCP_TXB0CTRL = 0x30, + MCP_TXB0SIDH = 0x31, + MCP_TXB0SIDL = 0x32, + MCP_TXB0EID8 = 0x33, + MCP_TXB0EID0 = 0x34, + MCP_TXB0DLC = 0x35, + MCP_TXB0DATA = 0x36, + MCP_TXB1CTRL = 0x40, + MCP_TXB1SIDH = 0x41, + MCP_TXB1SIDL = 0x42, + MCP_TXB1EID8 = 0x43, + MCP_TXB1EID0 = 0x44, + MCP_TXB1DLC = 0x45, + MCP_TXB1DATA = 0x46, + MCP_TXB2CTRL = 0x50, + MCP_TXB2SIDH = 0x51, + MCP_TXB2SIDL = 0x52, + MCP_TXB2EID8 = 0x53, + MCP_TXB2EID0 = 0x54, + MCP_TXB2DLC = 0x55, + MCP_TXB2DATA = 0x56, + MCP_RXB0CTRL = 0x60, + MCP_RXB0SIDH = 0x61, + MCP_RXB0SIDL = 0x62, + MCP_RXB0EID8 = 0x63, + MCP_RXB0EID0 = 0x64, + MCP_RXB0DLC = 0x65, + MCP_RXB0DATA = 0x66, + MCP_RXB1CTRL = 0x70, + MCP_RXB1SIDH = 0x71, + MCP_RXB1SIDL = 0x72, + MCP_RXB1EID8 = 0x73, + MCP_RXB1EID0 = 0x74, + MCP_RXB1DLC = 0x75, + MCP_RXB1DATA = 0x76 + }; + + static const uint32_t SPI_CLOCK = 10000000; // 10MHz + + static const int N_TXBUFFERS = 3; + static const int N_RXBUFFERS = 2; + + static const struct TXBn_REGS { + REGISTER CTRL; + REGISTER SIDH; + REGISTER DATA; + } TXB[N_TXBUFFERS]; + + static const struct RXBn_REGS { + REGISTER CTRL; + REGISTER SIDH; + REGISTER DATA; + CANINTF CANINTF_RXnIF; + } RXB[N_RXBUFFERS]; + + uint8_t SPICS; + + private: + + void startSPI(); + void endSPI(); + + ERROR setMode(const CANCTRL_REQOP_MODE mode); + + uint8_t readRegister(const REGISTER reg); + void readRegisters(const REGISTER reg, uint8_t values[], const uint8_t n); + void setRegister(const REGISTER reg, const uint8_t value); + void setRegisters(const REGISTER reg, const uint8_t values[], const uint8_t n); + void modifyRegister(const REGISTER reg, const uint8_t mask, const uint8_t data); + + void prepareId(uint8_t *buffer, const bool ext, const uint32_t id); + + public: + MCP2515(const uint8_t _CS); + ERROR reset(void); + ERROR setConfigMode(); + ERROR setListenOnlyMode(); + ERROR setSleepMode(); + ERROR setLoopbackMode(); + ERROR setNormalMode(); + ERROR setBitrate(const CAN_SPEED canSpeed); + ERROR setBitrate(const CAN_SPEED canSpeed, const CAN_CLOCK canClock); + ERROR setFilterMask(const MASK num, const bool ext, const uint32_t ulData); + ERROR setFilter(const RXF num, const bool ext, const uint32_t ulData); + ERROR sendMessage(const TXBn txbn, const struct can_frame *frame); + ERROR sendMessage(const struct can_frame *frame); + ERROR readMessage(const RXBn rxbn, struct can_frame *frame); + ERROR readMessage(struct can_frame *frame); + bool checkReceive(void); + bool checkError(void); + uint8_t getErrorFlags(void); + void clearRXnOVRFlags(void); + uint8_t getInterrupts(void); + uint8_t getInterruptMask(void); + void clearInterrupts(void); + void clearTXInterrupts(void); + uint8_t getStatus(void); + void clearRXnOVR(void); + void clearMERR(); +}; + +#endif diff --git a/tasmota/i18n.h b/tasmota/i18n.h index 6bf33d6a15a7..e72f1a4916bd 100644 --- a/tasmota/i18n.h +++ b/tasmota/i18n.h @@ -859,6 +859,8 @@ const char HTTP_SNS_COLOR_GREEN[] PROGMEM = "{s}%s " D_COLOR_GREEN "{ const char HTTP_SNS_COLOR_BLUE[] PROGMEM = "{s}%s " D_COLOR_BLUE "{m}%u " "{e}"; const char HTTP_SNS_MILLILITERS[] PROGMEM = "{s}%s " D_VOLUME "{m}%s " D_UNIT_MILLILITERS "{e}"; const char HTTP_SNS_GAS[] PROGMEM = "{s}%s " D_GAS "{m}%d " D_UNIT_PERCENT "LEL{e}"; +const char HTTP_SNS_SOC[] PROGMEM = "{s}%s " D_SOC "{m}%d " D_UNIT_PERCENT "{e}"; +const char HTTP_SNS_SOH[] PROGMEM = "{s}%s " D_SOH "{m}%d " D_UNIT_PERCENT "{e}"; #endif // USE_WEBSERVER #endif // _I18N_H_ diff --git a/tasmota/language/af_AF.h b/tasmota/language/af_AF.h index f60cd0444939..7085f0b56e55 100644 --- a/tasmota/language/af_AF.h +++ b/tasmota/language/af_AF.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Gewig" #define D_WARMLIGHT "Warm" #define D_WEB_SERVER "Webbediener" +#define D_SOC "Laai kondisie" +#define D_SOH "Laai vermoeë" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "WAARSKUWING Hierdie weergawe ondersteun nie aanhoudende instellings nie" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/bg_BG.h b/tasmota/language/bg_BG.h index 33a4b3e26632..c5df5e16cb3a 100644 --- a/tasmota/language/bg_BG.h +++ b/tasmota/language/bg_BG.h @@ -830,6 +830,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/cs_CZ.h b/tasmota/language/cs_CZ.h index e3fea7bebd88..4abba8442805 100644 --- a/tasmota/language/cs_CZ.h +++ b/tasmota/language/cs_CZ.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Hmotnost" #define D_WARMLIGHT "Teplé světlo" #define D_WEB_SERVER "Web Server" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "UPOZORNĚNÍ Tato verze nepodporuje trvalé nastavení" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/de_DE.h b/tasmota/language/de_DE.h index 11cc974568b3..d93f1135a98c 100644 --- a/tasmota/language/de_DE.h +++ b/tasmota/language/de_DE.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Gewicht" #define D_WARMLIGHT "warm" #define D_WEB_SERVER "Web-Server" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ACHTUNG: Diese Version unterstützt keine persistenten Einstellungen" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Puls" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/el_GR.h b/tasmota/language/el_GR.h index 8f8e1897412a..d6254a5c856c 100644 --- a/tasmota/language/el_GR.h +++ b/tasmota/language/el_GR.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Βάρος" #define D_WARMLIGHT "Θερμό" #define D_WEB_SERVER "Διακομιστής Web" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ΠΡΟΕΙΔΟΠΟΙΗΣΗ Αυτή η έκδοση δεν αποθηκεύει τις ρυθμίσεις" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/en_GB.h b/tasmota/language/en_GB.h index ba101fd4e152..8ab8fcae5c23 100644 --- a/tasmota/language/en_GB.h +++ b/tasmota/language/en_GB.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Weight" #define D_WARMLIGHT "Warm" #define D_WEB_SERVER "Web Server" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "WARNING This version does not support persistent settings" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/es_ES.h b/tasmota/language/es_ES.h index 7f73934d0ce5..cc5afd018b75 100644 --- a/tasmota/language/es_ES.h +++ b/tasmota/language/es_ES.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Peso" #define D_WARMLIGHT "Cálida" #define D_WEB_SERVER "Servidor Web" +#define D_SOC "Estado de Carga" +#define D_SOH "Estado de Salud" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "Cuidado, esta versión no guarda los cambios" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "Cruce por cero" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/fr_FR.h b/tasmota/language/fr_FR.h index cedf77d3106c..6eb7e9e39c95 100644 --- a/tasmota/language/fr_FR.h +++ b/tasmota/language/fr_FR.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Poids" #define D_WARMLIGHT "Chaud" #define D_WEB_SERVER "Serveur web" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ATTENTION Cette version ne supporte pas les réglages persistants" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/fy_NL.h b/tasmota/language/fy_NL.h index 9ba2b4dafdea..50f46a471c9d 100644 --- a/tasmota/language/fy_NL.h +++ b/tasmota/language/fy_NL.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Gewicht" #define D_WARMLIGHT "Waarm" #define D_WEB_SERVER "Webserver" +#define D_SOC "State of Charge" +#define D_SOH "State of Charge" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "WARSKOGING Dizze ferzje bewarret gjin ynstellings" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/he_HE.h b/tasmota/language/he_HE.h index 75fb110f9506..dc35190ee763 100644 --- a/tasmota/language/he_HE.h +++ b/tasmota/language/he_HE.h @@ -198,6 +198,8 @@ #define D_WEIGHT "משקל" #define D_WARMLIGHT "חום" #define D_WEB_SERVER "Web שרת" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "אזהרה גרסה זו אינה תומכת בהגדרות קבועות" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/hu_HU.h b/tasmota/language/hu_HU.h index f0cf4d6c7872..95a6599f283e 100644 --- a/tasmota/language/hu_HU.h +++ b/tasmota/language/hu_HU.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Tömeg" #define D_WARMLIGHT "Meleg fény" #define D_WEB_SERVER "Webszerver" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "VIGYÁZZ! Ez a verzió nem támogat tartós beállításokat" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/it_IT.h b/tasmota/language/it_IT.h index c73612508beb..021c2eacc0da 100644 --- a/tasmota/language/it_IT.h +++ b/tasmota/language/it_IT.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Peso" #define D_WARMLIGHT "Calda" #define D_WEB_SERVER "Server web" +#define D_SOC "Stato di Carica" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ATTENZIONE Questa versione non supporta il salvataggio delle impostazioni" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "Impulsi ZC" #define D_SENSOR_HALLEFFECT "Effetto hall" #define D_SENSOR_EPD_DATA "EPD - Dati" +#define D_SENSOR_MCP2515_CS "MCP2515 - CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/ko_KO.h b/tasmota/language/ko_KO.h index 07bf79d50fa2..c7683ecf7f08 100644 --- a/tasmota/language/ko_KO.h +++ b/tasmota/language/ko_KO.h @@ -198,6 +198,8 @@ #define D_WEIGHT "무게" #define D_WARMLIGHT "따뜻하게" #define D_WEB_SERVER "웹 서버" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "경고: 이 버전은 영구 설정을 지원하지 않습니다" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/nl_NL.h b/tasmota/language/nl_NL.h index e998fda83c27..f2830c299e68 100644 --- a/tasmota/language/nl_NL.h +++ b/tasmota/language/nl_NL.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Gewicht" #define D_WARMLIGHT "Warm" #define D_WEB_SERVER "Webserver" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "WAARSCHUWING Deze versie bewaart geen instellingen" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/pl_PL.h b/tasmota/language/pl_PL.h index a60f35d3bf47..5086423f0482 100644 --- a/tasmota/language/pl_PL.h +++ b/tasmota/language/pl_PL.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Waga" #define D_WARMLIGHT "Temperatura światła" #define D_WEB_SERVER "Serwer Web" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "UWAGA Ta wersja nie obsługuje zapisu ustawień" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "Efekt Halla" #define D_SENSOR_EPD_DATA "EPD Dane" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/pt_BR.h b/tasmota/language/pt_BR.h index 7b52aa2847b4..0233c77b3a71 100644 --- a/tasmota/language/pt_BR.h +++ b/tasmota/language/pt_BR.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Peso" #define D_WARMLIGHT "Luz quente" #define D_WEB_SERVER "Servidor WEB" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "AVISO: esta versão não supporta configurações persistentes" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "Efeito Hall" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/pt_PT.h b/tasmota/language/pt_PT.h index 6ddcfa2c8b66..df17497a682f 100644 --- a/tasmota/language/pt_PT.h +++ b/tasmota/language/pt_PT.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Peso" #define D_WARMLIGHT "Luz Quente" #define D_WEB_SERVER "Servidor WEB" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "AVISO esta versão não supporta configurações persistentes" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "Efeito Hall" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/ro_RO.h b/tasmota/language/ro_RO.h index ec5512332e99..89ae6aaecc3e 100644 --- a/tasmota/language/ro_RO.h +++ b/tasmota/language/ro_RO.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Greutate" #define D_WARMLIGHT "Cald" #define D_WEB_SERVER "Server Web" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ATENȚIE Această versiune nu suportă setări permanente" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/ru_RU.h b/tasmota/language/ru_RU.h index dd1f51026064..5574d26625ee 100644 --- a/tasmota/language/ru_RU.h +++ b/tasmota/language/ru_RU.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Weight" #define D_WARMLIGHT "Тепло" #define D_WEB_SERVER "Web сервер" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ПРЕДУПРЕЖДЕНИЕ Эта версия не поддерживает персистентные настройки" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "А" diff --git a/tasmota/language/sk_SK.h b/tasmota/language/sk_SK.h index 2ff2780503cc..67330366b7e2 100644 --- a/tasmota/language/sk_SK.h +++ b/tasmota/language/sk_SK.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Hmotnosť" #define D_WARMLIGHT "Teplé svetlo" #define D_WEB_SERVER "Web Server" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "UPOZORNENIE Táto verzia nepodporuje trvalé nastavenia" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/sv_SE.h b/tasmota/language/sv_SE.h index 828b13f4af52..cd9b6ee695fe 100644 --- a/tasmota/language/sv_SE.h +++ b/tasmota/language/sv_SE.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Vikt" #define D_WARMLIGHT "Varm" #define D_WEB_SERVER "Webbserver" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "VARNING Denna version supporterar inte beständiga inställningar" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/tr_TR.h b/tasmota/language/tr_TR.h index 0e4e837c1040..05a5202e0060 100644 --- a/tasmota/language/tr_TR.h +++ b/tasmota/language/tr_TR.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Weight" #define D_WARMLIGHT "Sıcak" #define D_WEB_SERVER "Web Sunucusu" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "UYARI Bu versiyon ayarların kalıcı olarak kaydedilmesine olanak sağlamıyor" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/uk_UA.h b/tasmota/language/uk_UA.h index d13226b9eae6..87ab4e37930d 100644 --- a/tasmota/language/uk_UA.h +++ b/tasmota/language/uk_UA.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Вага" #define D_WARMLIGHT "Тепло" #define D_WEB_SERVER "Web сервер" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ПОПЕРЕДЖЕННЯ! Ця версія не підтримує збереження налаштувань" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "А" diff --git a/tasmota/language/vi_VN.h b/tasmota/language/vi_VN.h index 0d406ab48dcf..1c290ddc26cf 100644 --- a/tasmota/language/vi_VN.h +++ b/tasmota/language/vi_VN.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Cân nặng" #define D_WARMLIGHT "Ấm" #define D_WEB_SERVER "Máy chủ Web" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "Cảnh báo phiên bản này không hỗ trợ các cài đặt vĩnh viễn" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/zh_CN.h b/tasmota/language/zh_CN.h index 19c8656f236b..8470c322ad61 100644 --- a/tasmota/language/zh_CN.h +++ b/tasmota/language/zh_CN.h @@ -198,6 +198,8 @@ #define D_WEIGHT "重量" #define D_WARMLIGHT "暖" #define D_WEB_SERVER "Web Server" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "警告:精简固件不支持持久保存设置" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "A" diff --git a/tasmota/language/zh_TW.h b/tasmota/language/zh_TW.h index a9abbd7609b0..317b1aad6b82 100644 --- a/tasmota/language/zh_TW.h +++ b/tasmota/language/zh_TW.h @@ -198,6 +198,8 @@ #define D_WEIGHT "重量" #define D_WARMLIGHT "暖光" #define D_WEB_SERVER "網頁伺服器" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "警告,這個版本並不支援將設定永久的儲存!" @@ -831,6 +833,7 @@ #define D_SENSOR_ZEROCROSS "ZC Pulse" #define D_SENSOR_HALLEFFECT "HallEffect" #define D_SENSOR_EPD_DATA "EPD Data" +#define D_SENSOR_MCP2515_CS "MCP2515 CS" // Units #define D_UNIT_AMPERE "安培" diff --git a/tasmota/tasmota_template.h b/tasmota/tasmota_template.h index c7b901e26948..1d5b449cdd16 100644 --- a/tasmota/tasmota_template.h +++ b/tasmota/tasmota_template.h @@ -158,6 +158,7 @@ enum UserSelectablePins { GPIO_MAX7219CLK, GPIO_MAX7219DIN, GPIO_MAX7219CS, // MAX7219 interface GPIO_TFMINIPLUS_TX, GPIO_TFMINIPLUS_RX, // TFmini Plus ToF sensor GPIO_ZEROCROSS, + GPIO_MCP2515_CS, // MCP2515 Chip Select #ifdef ESP32 GPIO_HALLEFFECT, GPIO_EPD_DATA, // Base connection EPD driver @@ -347,6 +348,7 @@ const char kSensorNames[] PROGMEM = D_SENSOR_MAX7219_CLK "|" D_SENSOR_MAX7219_DIN "|" D_SENSOR_MAX7219_CS "|" D_SENSOR_TFMINIPLUS_TX "|" D_SENSOR_TFMINIPLUS_RX "|" D_SENSOR_ZEROCROSS "|" + D_SENSOR_MCP2515_CS "|" #ifdef ESP32 D_SENSOR_HALLEFFECT "|" D_SENSOR_EPD_DATA "|" @@ -459,6 +461,9 @@ const uint16_t kGpioNiceList[] PROGMEM = { #ifdef USE_SDCARD AGPIO(GPIO_SDCARD_CS), #endif // USE_SDCARD +#ifdef USE_MCP2515 + AGPIO(GPIO_MCP2515_CS) +#endif // USE_MCP2515 #endif // USE_SPI AGPIO(GPIO_SSPI_MISO), // Software SPI Master Input Client Output diff --git a/tasmota/xsns_89_mcp2515.ino b/tasmota/xsns_89_mcp2515.ino new file mode 100644 index 000000000000..2fe3ea9f67fb --- /dev/null +++ b/tasmota/xsns_89_mcp2515.ino @@ -0,0 +1,189 @@ +/* + xsns_89_mcp2515.ino - MCP2515 CAN bus support for Tasmota + + Copyright (C) 2021 Marius Bezuidenhout + + 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_SPI +#ifdef USE_MCP2515 +/*********************************************************************************************\ + * MCP2515 - Microchip CAN controller + * + * Connections: + * MCP2515 ESP8266 Tasmota + * ------- -------------- ---------- + * INT not used + * SCK GPIO14 SPI CLK + * SI GPIO13 SPI MOSI + * SO GPIO12 SPI MISO + * CS GPIO0..5,15,16 MCP2515 + * Gnd Gnd + * VCC Vin/5V +\*********************************************************************************************/ + +#define XSNS_89 89 + +#include "mcp2515.h" + +// set defaults if not defined +#ifndef MCP2515_BITRATE + #define MCP2515_BITRATE CAN_500KBPS +#endif + +#ifndef MCP2515_CLOCK + #define MCP2515_CLOCK MCP_8MHZ +#endif + +#ifndef MCP2515_MAX_MSG + #define MCP2515_MAX_MSG 14 +#endif + +#ifndef MCP2515_BMS_CLIENT + #define MCP2515_BMS_CLIENT + // Look for Freedom Won BMS data in CAN message + #ifndef MCP2515_BMS_FREEDWON + #define MCP2515_BMS_FREEDWON + #endif // MCP2515_BMS_FREEDWON +#endif // MCP2515_BMS_CLIENT + +#ifdef MCP2515_BMS_CLIENT + struct BMS_Struct { + uint16_t stateOfCharge; + uint16_t stateOfHealth; + float battVoltage; + float battMilliAmp; + float battTemp; + char name[17]; +} bms; +#endif + +int8_t mcp2515_init_status = 1; + +struct can_frame canMsg; +MCP2515 mcp2515; + +void MCP2515_Init(void) { + mcp2515 = new MCP2515(5); + if (MCP2515::ERROR_OK != mcp2515.reset()) { + AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to reset module")); + mcp2515_init_status = 0; + } + + if (MCP2515::ERROR_OK != mcp2515.setBitrate(CAN_500KBPS, MCP_8MHZ)) { + AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to set module bitrate"); + mcp2515_init_status = 0; + } + + if (mcp2515_init_status && MCP2515::ERROR_OK != mcp2515.setNormalMode()) { + AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to set normal mode"); + mcp2515_init_status = 0; + } +} + +void MCP2515_Read() { + uint8_t nCounter = 0; + while (mcp2515.checkReceive() && nCounter <= MCP2515_MAX_MSG) { + nCounter++; + if (mcp2515.readMessage(&canMsg) == MCP2515::ERROR_OK) { +#ifdef MCP2515_BMS_CLIENT + #ifdef MCP2515_BMS_FREEDWON + switch(canMsg.can_id) { + case 0x355: + bms.stateOfCharge = canMsg.data[1] << 8 + canMsg.data[0]; + bms.stateOfHealth = canMsg.data[3] << 8 + canMsg.data[2]; + break; + case 0x356: + bms.battVoltage = (canMsg.data[1] << 8 + canMsg.data[0])/100; + bms.battMilliAmp = (canMsg.data[3] << 8 + canMsg.data[2])*100; + bms.battTemp = ConvertTemp((canMsg.data[5] << 8 + canMsg.data[4])/10); // Convert to fahrenheit if SetOpion8 is set + break; + case 0x370: + case 0x371: + for(int i = 0; i < canMsg.can_dlc; i++) { + bms.name[i + (8 * canMsg.can_id & 0x1)] = canMsg.data[i]; // If can_id is 0x371 then fill from byte 8 onwards + } + bms.name[16] = 0; // Ensure that name is null terminated + break; + default: + String canMsg; + for(int i = 0; i < canMsg.can_dlc; i++) { + canMsg += String(canMsg.data[i], HEX); + } + AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Received message 0x%s from ID 0x%X", canMsg, canMsg.can_id); + break; + } + #endif // MCP2515_BMS_FREEDWON +#endif // MCP2515_BMS_CLIENT + } else if(mcp2515.checkError()) { + uint8_t errFlags = mcp2515.getErrorFlags(); + mcp2515.clearRXnOVRFlags(); + AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Received error %d", errFlags); + break; + } + } +} + +void MCP2515_Show(bool Json) { + if (Json) { +#ifdef MCP2515_BMS_CLIENT + ResponseAppend_P(PSTR(",\"MCP2515\":{\"SOC\":%d,\"SOH\":%d}"), bms.stateOfCharge, bms.stateOfHealth); +#endif // MCP2515_BMS_CLIENT + +#ifdef USE_WEBSERVER + } else { + #ifdef MCP2515_BMS_CLIENT + WSContentSend_PD(HTTP_SNS_SOC, bms.name, bms.stateOfCharge); + WSContentSend_PD(HTTP_SNS_SOH, bms.name, bms.stateOfHealth); + WSContentSend_Voltage(bms.name, bms.battVoltage); + WSContentSend_CurrentMA(bms.name, bms.battMilliAmp); + WSContentSend_Temp(bms.name, bms.battTemp); + #endif // MCP2515_BMS_CLIENT +#endif // USE_WEBSERVER + } +} + +/*********************************************************************************************\ + * Interface +\*********************************************************************************************/ + +bool Xsns89(uint8_t function) +{ + bool result = false; + + if (PinUsed(GPIO_MCP2515_CS, GPIO_ANY) && TasmotaGlobal.spi_enabled) { + switch (function) { + case FUNC_INIT: + MCP2515_Init(); + break; + case FUNC_EVERY_SECOND: + MCP2515_Read(); + break; + case FUNC_JSON_APPEND: + MCP2515_Show(1); + break; + +#ifdef USE_WEBSERVER + case FUNC_WEB_SENSOR: + MCP2515_Show(0); + break; +#endif // USE_WEBSERVER + } + } + return result; +} + +#endif // USE_MCP2515 +#endif // USE_SPI \ No newline at end of file From f14f86f6fe300470ae323bf72e8308428d0589db Mon Sep 17 00:00:00 2001 From: Marius Bezuidenhout Date: Mon, 12 Jul 2021 17:47:04 +0200 Subject: [PATCH 2/8] Receiving SOC, SOH and Product name --- tasmota/support_tasmota.ino | 3 +- tasmota/tasmota_template.h | 4 +- tasmota/xsns_89_mcp2515.ino | 138 ++++++++++++++++++++++++------------ 3 files changed, 96 insertions(+), 49 deletions(-) diff --git a/tasmota/support_tasmota.ino b/tasmota/support_tasmota.ino index c519fc1a5117..7e55e804cf29 100644 --- a/tasmota/support_tasmota.ino +++ b/tasmota/support_tasmota.ino @@ -1765,7 +1765,8 @@ void GpioInit(void) ValidSpiPinUsed(GPIO_ST7789_DC) || // ST7789 CS may be omitted so chk DC too ValidSpiPinUsed(GPIO_ST7789_CS) || (ValidSpiPinUsed(GPIO_SSD1331_CS) && ValidSpiPinUsed(GPIO_SSD1331_DC)) || - ValidSpiPinUsed(GPIO_SDCARD_CS) + ValidSpiPinUsed(GPIO_SDCARD_CS) || + ValidSpiPinUsed(GPIO_MCP2515_CS) ); // If SPI_CS and/or SPI_DC is used they must be valid TasmotaGlobal.spi_enabled = (valid_cs) ? SPI_MOSI_MISO : SPI_NONE; diff --git a/tasmota/tasmota_template.h b/tasmota/tasmota_template.h index 1d5b449cdd16..9a97949f9e3f 100644 --- a/tasmota/tasmota_template.h +++ b/tasmota/tasmota_template.h @@ -158,7 +158,7 @@ enum UserSelectablePins { GPIO_MAX7219CLK, GPIO_MAX7219DIN, GPIO_MAX7219CS, // MAX7219 interface GPIO_TFMINIPLUS_TX, GPIO_TFMINIPLUS_RX, // TFmini Plus ToF sensor GPIO_ZEROCROSS, - GPIO_MCP2515_CS, // MCP2515 Chip Select + GPIO_MCP2515_CS, // MCP2515 Chip Select #ifdef ESP32 GPIO_HALLEFFECT, GPIO_EPD_DATA, // Base connection EPD driver @@ -462,7 +462,7 @@ const uint16_t kGpioNiceList[] PROGMEM = { AGPIO(GPIO_SDCARD_CS), #endif // USE_SDCARD #ifdef USE_MCP2515 - AGPIO(GPIO_MCP2515_CS) + AGPIO(GPIO_MCP2515_CS), #endif // USE_MCP2515 #endif // USE_SPI diff --git a/tasmota/xsns_89_mcp2515.ino b/tasmota/xsns_89_mcp2515.ino index 2fe3ea9f67fb..59d40bfc85a6 100644 --- a/tasmota/xsns_89_mcp2515.ino +++ b/tasmota/xsns_89_mcp2515.ino @@ -47,8 +47,8 @@ #define MCP2515_CLOCK MCP_8MHZ #endif -#ifndef MCP2515_MAX_MSG - #define MCP2515_MAX_MSG 14 +#ifndef MCP2515_MAX_FRAMES + #define MCP2515_MAX_FRAMES 14 #endif #ifndef MCP2515_BMS_CLIENT @@ -60,11 +60,11 @@ #endif // MCP2515_BMS_CLIENT #ifdef MCP2515_BMS_CLIENT - struct BMS_Struct { +struct BMS_Struct { uint16_t stateOfCharge; uint16_t stateOfHealth; float battVoltage; - float battMilliAmp; + float battAmp = 0; float battTemp; char name[17]; } bms; @@ -72,66 +72,110 @@ int8_t mcp2515_init_status = 1; -struct can_frame canMsg; -MCP2515 mcp2515; +struct can_frame canFrame; +MCP2515 *mcp2515 = nullptr; + +char c2h(char c) +{ + return "0123456789ABCDEF"[0x0F & (unsigned char)c]; +} void MCP2515_Init(void) { mcp2515 = new MCP2515(5); - if (MCP2515::ERROR_OK != mcp2515.reset()) { + if (MCP2515::ERROR_OK != mcp2515->reset()) { AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to reset module")); mcp2515_init_status = 0; } - if (MCP2515::ERROR_OK != mcp2515.setBitrate(CAN_500KBPS, MCP_8MHZ)) { - AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to set module bitrate"); + if (MCP2515::ERROR_OK != mcp2515->setBitrate(MCP2515_BITRATE, MCP2515_CLOCK)) { + AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to set module bitrate")); mcp2515_init_status = 0; } - if (mcp2515_init_status && MCP2515::ERROR_OK != mcp2515.setNormalMode()) { - AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to set normal mode"); + if (mcp2515_init_status && MCP2515::ERROR_OK != mcp2515->setNormalMode()) { + AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to set normal mode")); mcp2515_init_status = 0; } } void MCP2515_Read() { uint8_t nCounter = 0; - while (mcp2515.checkReceive() && nCounter <= MCP2515_MAX_MSG) { - nCounter++; - if (mcp2515.readMessage(&canMsg) == MCP2515::ERROR_OK) { + bool checkRcv; + if(mcp2515_init_status) { + + checkRcv = mcp2515->checkReceive(); + + while (checkRcv && nCounter <= MCP2515_MAX_FRAMES) { + mcp2515->checkReceive(); + nCounter++; + if (mcp2515->readMessage(&canFrame) == MCP2515::ERROR_OK) { #ifdef MCP2515_BMS_CLIENT #ifdef MCP2515_BMS_FREEDWON - switch(canMsg.can_id) { - case 0x355: - bms.stateOfCharge = canMsg.data[1] << 8 + canMsg.data[0]; - bms.stateOfHealth = canMsg.data[3] << 8 + canMsg.data[2]; - break; - case 0x356: - bms.battVoltage = (canMsg.data[1] << 8 + canMsg.data[0])/100; - bms.battMilliAmp = (canMsg.data[3] << 8 + canMsg.data[2])*100; - bms.battTemp = ConvertTemp((canMsg.data[5] << 8 + canMsg.data[4])/10); // Convert to fahrenheit if SetOpion8 is set - break; - case 0x370: - case 0x371: - for(int i = 0; i < canMsg.can_dlc; i++) { - bms.name[i + (8 * canMsg.can_id & 0x1)] = canMsg.data[i]; // If can_id is 0x371 then fill from byte 8 onwards - } - bms.name[16] = 0; // Ensure that name is null terminated - break; - default: - String canMsg; - for(int i = 0; i < canMsg.can_dlc; i++) { - canMsg += String(canMsg.data[i], HEX); - } - AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Received message 0x%s from ID 0x%X", canMsg, canMsg.can_id); - break; - } + switch(canFrame.can_id) { + // Charge/Discharge parameters + case 0x351: + break; + // State of Charge/Health + case 0x355: + if(6 >= canFrame.can_dlc) { + bms.stateOfCharge = (canFrame.data[1] << 8) + canFrame.data[0]; + bms.stateOfHealth = (canFrame.data[3] << 8) + canFrame.data[2]; + } else { + AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Unexpected length (%d) for ID 0x355"), canFrame.can_dlc); + } + break; + // Voltage/Current/Temperature + case 0x356: + if(6 >= canFrame.can_dlc) { + bms.battVoltage = (float)((canFrame.data[1] << 8) + canFrame.data[0])/100; + bms.battAmp = (float)((canFrame.data[3] << 8) + canFrame.data[2])/10; + bms.battTemp = ConvertTemp((float)((canFrame.data[5] << 8) + canFrame.data[4])/10); // Convert to fahrenheit if SetOpion8 is set + } else { + AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Unexpected length (%d) for ID 0x356"), canFrame.can_dlc); + } + break; + // Battery / BMS name + case 0x370: + case 0x371: + for(int i = 0; i < canFrame.can_dlc; i++) { + uint8_t nameStrPos = i + ((canFrame.can_id & 0x1) * 8); // If can_id is 0x371 then fill from byte 8 onwards + bms.name[nameStrPos] = canFrame.data[i]; + } + bms.name[16] = 0; // Ensure that name is null terminated + break; + // Modules status + case 0x372: + // Min. cell voltage id string + case 0x374: + // Min. cell temperature id string + case 0x376: + // Installed capacity + case 0x379: + // Serial number + case 0x380: + case 0x381: + break; + default: + char canMsg[17]; + canMsg[0] = 0; + for(int i = 0; i < canFrame.can_dlc; i++) { + canMsg[i*2] = c2h(canFrame.data[i]>>4); + canMsg[i*2+1] = c2h(canFrame.data[i]); + } + if(canFrame.can_dlc > 0) { + canMsg[(canFrame.can_dlc - 1) * 2 + 2] = 0; + } + AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Received message 0x%s from ID 0x%X"), canMsg, (uint32_t)canFrame.can_id); + break; + } #endif // MCP2515_BMS_FREEDWON #endif // MCP2515_BMS_CLIENT - } else if(mcp2515.checkError()) { - uint8_t errFlags = mcp2515.getErrorFlags(); - mcp2515.clearRXnOVRFlags(); - AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Received error %d", errFlags); - break; + } else if(mcp2515->checkError()) { + uint8_t errFlags = mcp2515->getErrorFlags(); + mcp2515->clearRXnOVRFlags(); + AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Received error %d"), errFlags); + break; + } } } } @@ -145,10 +189,12 @@ void MCP2515_Show(bool Json) { #ifdef USE_WEBSERVER } else { #ifdef MCP2515_BMS_CLIENT + char ampStr[6]; + dtostrf(bms.battAmp, 5, 1, ampStr); WSContentSend_PD(HTTP_SNS_SOC, bms.name, bms.stateOfCharge); WSContentSend_PD(HTTP_SNS_SOH, bms.name, bms.stateOfHealth); WSContentSend_Voltage(bms.name, bms.battVoltage); - WSContentSend_CurrentMA(bms.name, bms.battMilliAmp); + WSContentSend_PD(HTTP_SNS_CURRENT, ampStr); WSContentSend_Temp(bms.name, bms.battTemp); #endif // MCP2515_BMS_CLIENT #endif // USE_WEBSERVER @@ -168,7 +214,7 @@ bool Xsns89(uint8_t function) case FUNC_INIT: MCP2515_Init(); break; - case FUNC_EVERY_SECOND: + case FUNC_EVERY_50_MSECOND: MCP2515_Read(); break; case FUNC_JSON_APPEND: From ad8b557becf3e9f1bc98fe9e3fd68b810c66953a Mon Sep 17 00:00:00 2001 From: Marius Bezuidenhout Date: Tue, 13 Jul 2021 16:49:40 +0200 Subject: [PATCH 3/8] First working version --- tasmota/xsns_89_mcp2515.ino | 78 ++++++++++++++++++++++++++++--------- 1 file changed, 59 insertions(+), 19 deletions(-) diff --git a/tasmota/xsns_89_mcp2515.ino b/tasmota/xsns_89_mcp2515.ino index 59d40bfc85a6..f6ca4b873dcc 100644 --- a/tasmota/xsns_89_mcp2515.ino +++ b/tasmota/xsns_89_mcp2515.ino @@ -51,6 +51,14 @@ #define MCP2515_MAX_FRAMES 14 #endif +#ifndef CAN_KEEP_ALIVE_SECS + #define CAN_KEEP_ALIVE_SECS 300 +#endif + +#ifndef MCP2515_TIMEOUT + #define MCP2515_TIMEOUT 10 +#endif + #ifndef MCP2515_BMS_CLIENT #define MCP2515_BMS_CLIENT // Look for Freedom Won BMS data in CAN message @@ -63,15 +71,16 @@ struct BMS_Struct { uint16_t stateOfCharge; uint16_t stateOfHealth; - float battVoltage; - float battAmp = 0; - float battTemp; + uint16_t battVoltage; // Div 100 + int16_t battAmp; // Div 10 + int16_t battTemp; // Div 10 char name[17]; } bms; #endif int8_t mcp2515_init_status = 1; +uint32_t lastFrameRecv = 0; struct can_frame canFrame; MCP2515 *mcp2515 = nullptr; @@ -96,12 +105,17 @@ void MCP2515_Init(void) { AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to set normal mode")); mcp2515_init_status = 0; } +#ifdef MCP2515_BMS_FREEDWON + // TODO: Filter CAN bus messages + //mcp2515->setFilterMask(); + //mcp2515->setFilter(); +#endif } void MCP2515_Read() { uint8_t nCounter = 0; bool checkRcv; - if(mcp2515_init_status) { + if (mcp2515_init_status) { checkRcv = mcp2515->checkReceive(); @@ -109,31 +123,37 @@ void MCP2515_Read() { mcp2515->checkReceive(); nCounter++; if (mcp2515->readMessage(&canFrame) == MCP2515::ERROR_OK) { + lastFrameRecv = TasmotaGlobal.uptime; #ifdef MCP2515_BMS_CLIENT #ifdef MCP2515_BMS_FREEDWON - switch(canFrame.can_id) { + switch (canFrame.can_id) { // Charge/Discharge parameters case 0x351: break; // State of Charge/Health case 0x355: - if(6 >= canFrame.can_dlc) { - bms.stateOfCharge = (canFrame.data[1] << 8) + canFrame.data[0]; - bms.stateOfHealth = (canFrame.data[3] << 8) + canFrame.data[2]; + if (6 >= canFrame.can_dlc) { + bms.stateOfCharge = (canFrame.data[1] << 8) | canFrame.data[0]; + bms.stateOfHealth = (canFrame.data[3] << 8) | canFrame.data[2]; } else { AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Unexpected length (%d) for ID 0x355"), canFrame.can_dlc); } break; // Voltage/Current/Temperature case 0x356: - if(6 >= canFrame.can_dlc) { - bms.battVoltage = (float)((canFrame.data[1] << 8) + canFrame.data[0])/100; - bms.battAmp = (float)((canFrame.data[3] << 8) + canFrame.data[2])/10; - bms.battTemp = ConvertTemp((float)((canFrame.data[5] << 8) + canFrame.data[4])/10); // Convert to fahrenheit if SetOpion8 is set + if (6 >= canFrame.can_dlc) { + bms.battVoltage = (canFrame.data[1] << 8) | canFrame.data[0]; + bms.battAmp = (canFrame.data[3] << 8) | canFrame.data[2]; + bms.battTemp = (canFrame.data[5] << 8) | canFrame.data[4]; // Convert to fahrenheit if SetOpion8 is set } else { AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Unexpected length (%d) for ID 0x356"), canFrame.can_dlc); } break; + // Manufacturer name + case 0x35E: + // Battery Model / Firmware version + case 0x35F: + break; // Battery / BMS name case 0x370: case 0x371: @@ -145,10 +165,16 @@ void MCP2515_Read() { break; // Modules status case 0x372: + // Min/Max cell voltage/temperature + case 0x373: // Min. cell voltage id string case 0x374: + // Max. cell voltage id string + case 0x375: // Min. cell temperature id string case 0x376: + // Max. cell temperature id string + case 0x377: // Installed capacity case 0x379: // Serial number @@ -162,7 +188,7 @@ void MCP2515_Read() { canMsg[i*2] = c2h(canFrame.data[i]>>4); canMsg[i*2+1] = c2h(canFrame.data[i]); } - if(canFrame.can_dlc > 0) { + if (canFrame.can_dlc > 0) { canMsg[(canFrame.can_dlc - 1) * 2 + 2] = 0; } AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Received message 0x%s from ID 0x%X"), canMsg, (uint32_t)canFrame.can_id); @@ -170,32 +196,46 @@ void MCP2515_Read() { } #endif // MCP2515_BMS_FREEDWON #endif // MCP2515_BMS_CLIENT - } else if(mcp2515->checkError()) { + } else if (mcp2515->checkError()) { uint8_t errFlags = mcp2515->getErrorFlags(); mcp2515->clearRXnOVRFlags(); AddLog(LOG_LEVEL_DEBUG, PSTR("MCP2515: Received error %d"), errFlags); break; } } + +#ifdef MCP2515_BMS_FREEDWON + if (!(TasmotaGlobal.uptime%CAN_KEEP_ALIVE_SECS) && TasmotaGlobal.uptime>60) { + canFrame.can_id = 0x305; + canFrame.can_dlc = 0; + if (MCP2515::ERROR_OK != mcp2515->sendMessage(&canFrame)) { + AddLog(LOG_LEVEL_ERROR, PSTR("MCP2515: Failed to send keep alive frame")); + } + } +#endif } } void MCP2515_Show(bool Json) { if (Json) { + if (lastFrameRecv > 0 && TasmotaGlobal.uptime - lastFrameRecv <= MCP2515_TIMEOUT ) { #ifdef MCP2515_BMS_CLIENT - ResponseAppend_P(PSTR(",\"MCP2515\":{\"SOC\":%d,\"SOH\":%d}"), bms.stateOfCharge, bms.stateOfHealth); + ResponseAppend_P(PSTR(",\"MCP2515\":{\"SOC\":%d,\"SOH\":%d}"), \ + bms.stateOfCharge, + bms.stateOfHealth + ); #endif // MCP2515_BMS_CLIENT - + } #ifdef USE_WEBSERVER } else { #ifdef MCP2515_BMS_CLIENT char ampStr[6]; - dtostrf(bms.battAmp, 5, 1, ampStr); + dtostrf((float(bms.battAmp) / 10), 5, 1, ampStr); WSContentSend_PD(HTTP_SNS_SOC, bms.name, bms.stateOfCharge); WSContentSend_PD(HTTP_SNS_SOH, bms.name, bms.stateOfHealth); - WSContentSend_Voltage(bms.name, bms.battVoltage); + WSContentSend_Voltage(bms.name, (float(bms.battVoltage) / 100)); WSContentSend_PD(HTTP_SNS_CURRENT, ampStr); - WSContentSend_Temp(bms.name, bms.battTemp); + WSContentSend_Temp(bms.name, ConvertTemp(float(bms.battTemp) / 10)); #endif // MCP2515_BMS_CLIENT #endif // USE_WEBSERVER } From c366ce067dd8c7560cee4a5e836de3ce6ac499ee Mon Sep 17 00:00:00 2001 From: Jason2866 <24528715+Jason2866@users.noreply.github.com> Date: Tue, 13 Jul 2021 18:07:58 +0200 Subject: [PATCH 4/8] Add missing entrys --- tasmota/language/bg_BG.h | 2 ++ 1 file changed, 2 insertions(+) diff --git a/tasmota/language/bg_BG.h b/tasmota/language/bg_BG.h index c5df5e16cb3a..2e55b828006b 100644 --- a/tasmota/language/bg_BG.h +++ b/tasmota/language/bg_BG.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Тегло" #define D_WARMLIGHT "Топла" #define D_WEB_SERVER "Уеб сървър" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ПРЕДУПРЕЖДЕНИЕ Тази версия не поддържа постоянни настройки" From 44e8b838cfae4f07b4e815fc51baf44caa63d3e4 Mon Sep 17 00:00:00 2001 From: Marius Bezuidenhout Date: Tue, 13 Jul 2021 20:18:11 +0200 Subject: [PATCH 5/8] Added missing translations --- tasmota/language/bg_BG.h | 2 ++ 1 file changed, 2 insertions(+) diff --git a/tasmota/language/bg_BG.h b/tasmota/language/bg_BG.h index c5df5e16cb3a..2e55b828006b 100644 --- a/tasmota/language/bg_BG.h +++ b/tasmota/language/bg_BG.h @@ -198,6 +198,8 @@ #define D_WEIGHT "Тегло" #define D_WARMLIGHT "Топла" #define D_WEB_SERVER "Уеб сървър" +#define D_SOC "State of Charge" +#define D_SOH "State of Health" // tasmota.ino #define D_WARNING_MINIMAL_VERSION "ПРЕДУПРЕЖДЕНИЕ Тази версия не поддържа постоянни настройки" From 0987489c0c53f7e810e3e4beb71004148169ad80 Mon Sep 17 00:00:00 2001 From: Marius Bezuidenhout Date: Wed, 14 Jul 2021 09:40:36 +0200 Subject: [PATCH 6/8] Fixed compile on ESP32 --- lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h | 1 + 1 file changed, 1 insertion(+) diff --git a/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h b/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h index a73d1500e578..84c4216b146f 100644 --- a/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h +++ b/lib/lib_div/arduino-mcp2515-1.0.1/mcp2515.h @@ -1,6 +1,7 @@ #ifndef _MCP2515_H_ #define _MCP2515_H_ +#include #include #include "can.h" From 425eb723054d970c935f8567420b4378d4999fd5 Mon Sep 17 00:00:00 2001 From: Marius Bezuidenhout Date: Wed, 14 Jul 2021 11:43:26 +0200 Subject: [PATCH 7/8] Changed sensor number to 90 --- tasmota/{xsns_89_mcp2515.ino => xsns_90_mcp2515.ino} | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) rename tasmota/{xsns_89_mcp2515.ino => xsns_90_mcp2515.ino} (99%) diff --git a/tasmota/xsns_89_mcp2515.ino b/tasmota/xsns_90_mcp2515.ino similarity index 99% rename from tasmota/xsns_89_mcp2515.ino rename to tasmota/xsns_90_mcp2515.ino index f6ca4b873dcc..8f37063a2eaf 100644 --- a/tasmota/xsns_89_mcp2515.ino +++ b/tasmota/xsns_90_mcp2515.ino @@ -34,7 +34,7 @@ * VCC Vin/5V \*********************************************************************************************/ -#define XSNS_89 89 +#define XSNS_90 90 #include "mcp2515.h" @@ -245,7 +245,7 @@ void MCP2515_Show(bool Json) { * Interface \*********************************************************************************************/ -bool Xsns89(uint8_t function) +bool Xsns90(uint8_t function) { bool result = false; From 5fec3b0ac7a8da2cf590cc4e128bd7d1984e9b10 Mon Sep 17 00:00:00 2001 From: Marius Bezuidenhout Date: Wed, 14 Jul 2021 11:54:54 +0200 Subject: [PATCH 8/8] Moved entries to end of list --- tasmota/tasmota_template.h | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/tasmota/tasmota_template.h b/tasmota/tasmota_template.h index ac67f8c76b00..6e0d6d33f8a0 100644 --- a/tasmota/tasmota_template.h +++ b/tasmota/tasmota_template.h @@ -158,7 +158,6 @@ enum UserSelectablePins { GPIO_MAX7219CLK, GPIO_MAX7219DIN, GPIO_MAX7219CS, // MAX7219 interface GPIO_TFMINIPLUS_TX, GPIO_TFMINIPLUS_RX, // TFmini Plus ToF sensor GPIO_ZEROCROSS, - GPIO_MCP2515_CS, // MCP2515 Chip Select #ifdef ESP32 GPIO_HALLEFFECT, GPIO_EPD_DATA, // Base connection EPD driver @@ -170,6 +169,7 @@ enum UserSelectablePins { GPIO_I2S_OUT_DATA, GPIO_I2S_OUT_CLK, GPIO_I2S_OUT_SLCT, GPIO_I2S_IN_DATA, GPIO_I2S_IN_CLK, GPIO_I2S_IN_SLCT, GPIO_INTERRUPT, + GPIO_MCP2515_CS, // MCP2515 Chip Select GPIO_SENSOR_END }; enum ProgramSelectablePins { @@ -348,7 +348,6 @@ const char kSensorNames[] PROGMEM = D_SENSOR_MAX7219_CLK "|" D_SENSOR_MAX7219_DIN "|" D_SENSOR_MAX7219_CS "|" D_SENSOR_TFMINIPLUS_TX "|" D_SENSOR_TFMINIPLUS_RX "|" D_SENSOR_ZEROCROSS "|" - D_SENSOR_MCP2515_CS "|" #ifdef ESP32 D_SENSOR_HALLEFFECT "|" D_SENSOR_EPD_DATA "|" @@ -360,6 +359,7 @@ const char kSensorNames[] PROGMEM = D_SENSOR_I2S_OUT_DATA "|" D_SENSOR_I2S_OUT_CLK "|" D_SENSOR_I2S_OUT_SLCT "|" D_SENSOR_I2S_IN_DATA "|" D_SENSOR_I2S_IN_CLK "|" D_SENSOR_I2S_IN_SLCT "|" D_SENSOR_INTERRUPT "|" + D_SENSOR_MCP2515_CS "|" ; const char kSensorNamesFixed[] PROGMEM =