diff --git a/DCCWaveformRMT.cpp b/DCCWaveformRMT.cpp
index 5153dc7..aa3c223 100644
--- a/DCCWaveformRMT.cpp
+++ b/DCCWaveformRMT.cpp
@@ -32,6 +32,14 @@ DCCWaveform DCCWaveform::progTrack(PREAMBLE_BITS_PROG, false);
RMTChannel *DCCWaveform::rmtMainChannel = NULL;
RMTChannel *DCCWaveform::rmtProgChannel = NULL;
+bool DCCWaveform::railcomPossible=false; // High accuracy only
+volatile bool DCCWaveform::railcomActive=false; // switched on by user
+volatile bool DCCWaveform::railcomDebug=false; // switched on by user
+volatile bool DCCWaveform::railcomSampleWindow=false; // true during packet transmit
+volatile byte DCCWaveform::railcomCutoutCounter=0; // cyclic cutout
+volatile byte DCCWaveform::railcomLastAddressHigh=0;
+volatile byte DCCWaveform::railcomLastAddressLow=0;
+
DCCWaveform::DCCWaveform(byte preambleBits, bool isMain) {
isMainTrack = isMain;
requiredPreambles = preambleBits;
diff --git a/EXRAILMacros.h b/EXRAILMacros.h
index ae13fe0..11ffdc1 100644
--- a/EXRAILMacros.h
+++ b/EXRAILMacros.h
@@ -24,6 +24,7 @@
#ifndef EXRAILMacros_H
#define EXRAILMacros_H
+#include "IODeviceList.h"
// remove normal code LCD & SERIAL macros (will be restored later)
#undef LCD
diff --git a/IODevice.h b/IODevice.h
index af62c6e..43c7216 100644
--- a/IODevice.h
+++ b/IODevice.h
@@ -560,6 +560,6 @@ protected:
};
-#include "IODeviceList.h"
+//#include "IODeviceList.h"
#endif // iodevice_h
diff --git a/IO_EncoderThrottle.cpp b/IO_EncoderThrottle.cpp
index e6019a8..b2218b4 100644
--- a/IO_EncoderThrottle.cpp
+++ b/IO_EncoderThrottle.cpp
@@ -24,6 +24,7 @@
*/
#include "IODevice.h"
+#include "IO_EncoderThrottle.h"
#include "DIAG.h"
#include "DCC.h"
diff --git a/IO_I2CRailcom.cpp b/IO_I2CRailcom.cpp
new file mode 100644
index 0000000..3c55c74
--- /dev/null
+++ b/IO_I2CRailcom.cpp
@@ -0,0 +1,303 @@
+ /*
+ * © 2024, Henk Kruisbrink & Chris Harlow. All rights reserved.
+ * © 2023, Neil McKechnie. All rights reserved.
+ *
+ * This file is part of DCC++EX API
+ *
+ * This 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.
+ *
+ * It 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 CommandStation. If not, see .
+ */
+
+/*
+ *
+ * Dec 2023, Added NXP SC16IS752 I2C Dual UART
+ * The SC16IS752 has 64 bytes TX & RX FIFO buffer
+ * First version without interrupts from I2C UART and only RX/TX are used, interrupts may not be
+ * needed as the RX Fifo holds the reply
+ *
+ * Jan 2024, Issue with using both UARTs simultaniously, the secod uart seems to work but the first transmit
+ * corrupt data. This need more analysis and experimenatation.
+ * Will push this driver to the dev branch with the uart fixed to 0
+ * Both SC16IS750 (single uart) and SC16IS752 (dual uart, but only uart 0 is enable)
+ *
+ * myHall.cpp configuration syntax:
+ *
+ * I2CRailcom::create(1st vPin, vPins, I2C address);
+ *
+ * myAutomation configuration
+ * HAL(I2CRailcom, 1st vPin, vPins, I2C address)
+ * Parameters:
+ * 1st vPin : First virtual pin that EX-Rail can control to play a sound, use PLAYSOUND command (alias of ANOUT)
+ * vPins : Total number of virtual pins allocated (2 vPins are supported, one for each UART)
+ * 1st vPin for UART 0, 2nd for UART 1
+ * I2C Address : I2C address of the serial controller, in 0x format
+ */
+
+#include "IODevice.h"
+#include "IO_I2CRailcom.h"
+#include "I2CManager.h"
+#include "DIAG.h"
+#include "DCCWaveform.h"
+#include "Railcom.h"
+
+// Debug and diagnostic defines, enable too many will result in slowing the driver
+#define DIAG_I2CRailcom
+
+ I2CRailcom::I2CRailcom(VPIN firstVpin, int nPins, I2CAddress i2cAddress){
+ _firstVpin = firstVpin;
+ _nPins = nPins;
+ _I2CAddress = i2cAddress;
+ _channelMonitors[0]=new Railcom(firstVpin);
+ if (nPins>1) _channelMonitors[1]=new Railcom(firstVpin+1);
+ addDevice(this);
+ }
+
+void I2CRailcom::create(VPIN firstVpin, int nPins, I2CAddress i2cAddress) {
+ if (nPins>2) nPins=2;
+ if (checkNoOverlap(firstVpin, nPins, i2cAddress))
+ new I2CRailcom(firstVpin, nPins, i2cAddress);
+ }
+
+ void I2CRailcom::_begin() {
+ I2CManager.setClock(1000000); // TODO do we need this?
+ I2CManager.begin();
+ auto exists=I2CManager.exists(_I2CAddress);
+ DIAG(F("I2CRailcom: %s UART%S detected"),
+ _I2CAddress.toString(), exists?F(""):F(" NOT"));
+ if (!exists) return;
+
+ _UART_CH=0;
+ Init_SC16IS752(); // Initialize UART0
+ if (_nPins>1) {
+ _UART_CH=1;
+ Init_SC16IS752(); // Initialize UART1
+ }
+
+ if (_deviceState==DEVSTATE_INITIALISING) _deviceState=DEVSTATE_NORMAL;
+ _display();
+ }
+
+
+ void I2CRailcom::_loop(unsigned long currentMicros) {
+ // Read responses from device
+ if (_deviceState!=DEVSTATE_NORMAL) return;
+
+ // return if in cutout or cutout very soon.
+ if (!DCCWaveform::isRailcomSampleWindow()) return;
+
+ // IF we have 2 channels, flip channels each loop
+ if (_nPins>1) _UART_CH=_UART_CH?0:1;
+
+ // have we read this cutout already?
+ auto cut=DCCWaveform::getRailcomCutoutCounter();
+ if (cutoutCounter[_UART_CH]==cut) return;
+ cutoutCounter[_UART_CH]=cut;
+
+ // Read incoming raw Railcom data, and process accordingly
+
+ auto inlength = UART_ReadRegister(REG_RXLV);
+
+ if (inlength> sizeof(_inbuf)) inlength=sizeof(_inbuf);
+ _inbuf[0]=0;
+ if (inlength>0) {
+ // Read data buffer from UART
+ _outbuf[0]=(byte)(REG_RHR << 3 | _UART_CH << 1);
+ I2CManager.read(_I2CAddress, _inbuf, inlength, _outbuf, 1);
+ }
+ // HK: Reset FIFO at end of read cycle
+ UART_WriteRegister(REG_FCR, 0x07,false);
+
+ // Ask Railcom to interpret the raw data
+ _channelMonitors[_UART_CH]->process(_inbuf,inlength);
+ }
+
+
+ void I2CRailcom::_display() {
+ DIAG(F("I2CRailcom: Configured on Vpins:%u-%u %S"), _firstVpin, _firstVpin+_nPins-1,
+ (_deviceState!=DEVSTATE_NORMAL) ? F("OFFLINE") : F(""));
+ }
+
+ // SC16IS752 functions
+ // Initialise SC16IS752 only for this channel
+ // First a software reset
+ // Enable FIFO and clear TX & RX FIFO
+ // Need to set the following registers
+ // IOCONTROL set bit 1 and 2 to 0 indicating that they are GPIO
+ // IODIR set all bit to 1 indicating al are output
+ // IOSTATE set only bit 0 to 1 for UART 0, or only bit 1 for UART 1 //
+ // LCR bit 7=0 divisor latch (clock division registers DLH & DLL, they store 16 bit divisor),
+ // WORD_LEN, STOP_BIT, PARITY_ENA and PARITY_TYPE
+ // MCR bit 7=0 clock divisor devide-by-1 clock input
+ // DLH most significant part of divisor
+ // DLL least significant part of divisor
+ //
+ // BAUD_RATE, WORD_LEN, STOP_BIT, PARITY_ENA and PARITY_TYPE have been defined and initialized
+ //
+ // Communication parameters 8 bit, No parity, 1 stopbit
+ static const uint8_t WORD_LEN = 0x03; // Value LCR bit 0,1
+ static const uint8_t STOP_BIT = 0x00; // Value LCR bit 2
+ static const uint8_t PARITY_ENA = 0x00; // Value LCR bit 3
+ static const uint8_t PARITY_TYPE = 0x00; // Value LCR bit 4
+ static const uint32_t BAUD_RATE = 250000;
+ static const uint8_t PRESCALER = 0x01; // Value MCR bit 7
+ static const unsigned long SC16IS752_XTAL_FREQ_RAILCOM = 16000000; // Baud rate for Railcom signal
+ static const uint16_t _divisor = (SC16IS752_XTAL_FREQ_RAILCOM / PRESCALER) / (BAUD_RATE * 16);
+
+ void I2CRailcom::Init_SC16IS752(){
+ if (_UART_CH==0) { // HK: Currently fixed on ch 0
+ // only reset on channel 0}
+ UART_WriteRegister(REG_IOCONTROL, 0x08,false); // UART Software reset
+ //_deviceState=DEVSTATE_INITIALISING; // ignores error during reset which seems normal. // HK: this line is moved to below
+ auto iocontrol_readback = UART_ReadRegister(REG_IOCONTROL);
+ if (iocontrol_readback == 0x00){
+ _deviceState=DEVSTATE_INITIALISING;
+ DIAG(F("I2CRailcom: %s SRESET readback: 0x%x"),_I2CAddress.toString(), iocontrol_readback);
+ } else {
+ DIAG(F("I2CRailcom: %s SRESET: 0x%x"),_I2CAddress.toString(), iocontrol_readback);
+ }
+ }
+ // HK:
+ // You write 0x08 to the IOCONTROL register, setting bit 3 (SRESET), as per datasheet 8.18:
+ // "Software Reset. A write to this bit will reset the device. Once the
+ // device is reset this bit is automatically set to logic 0"
+ // So you can not readback the val you have written as this has changed.
+ // I've added an extra UART_ReadRegister(REG_IOCONTROL) and check if the return value is 0x00
+ // then set _deviceState=DEVSTATE_INITIALISING;
+
+
+ // HK: only do clear FIFO at end of Init_SC16IS752
+ //UART_WriteRegister(REG_FCR, 0x07,false); // Reset FIFO, clear RX & TX FIFO (write only)
+
+ UART_WriteRegister(REG_MCR, 0x00); // Set MCR to all 0, includes Clock divisor
+
+ //UART_WriteRegister(REG_LCR, 0x80); // Divisor latch enabled
+
+ UART_WriteRegister(REG_LCR, 0x80 | WORD_LEN | STOP_BIT | PARITY_ENA | PARITY_TYPE); // Divisor latch enabled and comm parameters set
+ UART_WriteRegister(REG_DLL, (uint8_t)_divisor); // Write DLL
+ UART_WriteRegister(REG_DLH, (uint8_t)(_divisor >> 8)); // Write DLH
+ auto lcr_readback = UART_ReadRegister(REG_LCR);
+ lcr_readback = lcr_readback & 0x7F;
+ UART_WriteRegister(REG_LCR, lcr_readback); // Divisor latch disabled
+
+ //UART_WriteRegister(REG_LCR, WORD_LEN | STOP_BIT | PARITY_ENA | PARITY_TYPE); // Divisor latch disabled
+
+ UART_WriteRegister(REG_FCR, 0x07,false); // Reset FIFO, clear RX & TX FIFO (write only)
+
+ #ifdef DIAG_I2CRailcom
+ // HK: Test to see if internal loopback works and if REG_RXLV increment to at least 0x01
+ // Set REG_MCR bit 4 to 1, Enable Loopback
+ UART_WriteRegister(REG_MCR, 0x10);
+ UART_WriteRegister(REG_THR, 0x88, false); // Send 0x88
+ auto inlen = UART_ReadRegister(REG_RXLV);
+ if (inlen == 0){
+ DIAG(F("I2CRailcom: Loopback test: %s/%d failed"),_I2CAddress.toString(), _UART_CH);
+ } else {
+ DIAG(F("Railcom: Loopback test: %s/%d RX Fifo lvl: 0x%x"),_I2CAddress.toString(), _UART_CH, inlen);
+ _outbuf[0]=(byte)(REG_RHR << 3 | _UART_CH << 1);
+ I2CManager.read(_I2CAddress, _inbuf, inlen, _outbuf, 1);
+ #ifdef DIAG_I2CRailcom_data
+ DIAG(F("Railcom: Loopback test: %s/%d RX FIFO Data"), _I2CAddress.toString(), _UART_CH);
+ for (int i = 0; i < inlen; i++){
+ DIAG(F("Railcom: Loopback data [0x%x]: 0x%x"), i, _inbuf[i]);
+ //DIAG(F("[0x%x]: 0x%x"), i, _inbuf[i]);
+ }
+ #endif
+ }
+ UART_WriteRegister(REG_MCR, 0x00); // Set REG_MCR back to 0x00
+ #endif
+
+ #ifdef DIAG_I2CRailcom
+ // Sent some data to check if UART baudrate is set correctly, check with logic analyzer on TX pin
+ UART_WriteRegister(REG_THR, 9, false);
+ DIAG(F("I2CRailcom: UART %s/%d Test TX = 0x09"),_I2CAddress.toString(), _UART_CH);
+ #endif
+
+ if (_deviceState==DEVSTATE_INITIALISING) {
+ DIAG(F("I2CRailcom: UART %d init complete"),_UART_CH);
+ }
+ // HK: final FIFO reset
+ UART_WriteRegister(REG_FCR, 0x07,false); // Reset FIFO, clear RX & TX FIFO (write only)
+
+ }
+
+
+
+
+ void I2CRailcom::UART_WriteRegister(uint8_t reg, uint8_t val, bool readback){
+ _outbuf[0] = (byte)( reg << 3 | _UART_CH << 1);
+ _outbuf[1]=val;
+ auto status=I2CManager.write(_I2CAddress, _outbuf, (uint8_t)2);
+ if(status!=I2C_STATUS_OK) {
+ DIAG(F("I2CRailcom: %s/%d write reg=0x%x,data=0x%x,I2Cstate=%d"),
+ _I2CAddress.toString(), _UART_CH, reg, val, status);
+ _deviceState=DEVSTATE_FAILED;
+ }
+ if (readback) { // Read it back to cross check
+ auto readback=UART_ReadRegister(reg);
+ if (readback!=val) {
+ DIAG(F("I2CRailcom readback: %s/%d reg:0x%x write=0x%x read=0x%x"),_I2CAddress.toString(),_UART_CH,reg,val,readback);
+ }
+ }
+ }
+
+
+ uint8_t I2CRailcom::UART_ReadRegister(uint8_t reg){
+ _outbuf[0] = (byte)(reg << 3 | _UART_CH << 1); // _outbuffer[0] has now UART_REG and UART_CH
+ _inbuf[0]=0;
+ auto status=I2CManager.read(_I2CAddress, _inbuf, 1, _outbuf, 1);
+ if (status!=I2C_STATUS_OK) {
+ DIAG(F("I2CRailcom read: %s/%d read reg=0x%x,I2Cstate=%d"),
+ _I2CAddress.toString(), _UART_CH, reg, status);
+ _deviceState=DEVSTATE_FAILED;
+ }
+ return _inbuf[0];
+ }
+
+// SC16IS752 General register set (from the datasheet)
+enum : uint8_t {
+ REG_RHR = 0x00, // FIFO Read
+ REG_THR = 0x00, // FIFO Write
+ REG_IER = 0x01, // Interrupt Enable Register R/W
+ REG_FCR = 0x02, // FIFO Control Register Write
+ REG_IIR = 0x02, // Interrupt Identification Register Read
+ REG_LCR = 0x03, // Line Control Register R/W
+ REG_MCR = 0x04, // Modem Control Register R/W
+ REG_LSR = 0x05, // Line Status Register Read
+ REG_MSR = 0x06, // Modem Status Register Read
+ REG_SPR = 0x07, // Scratchpad Register R/W
+ REG_TCR = 0x06, // Transmission Control Register R/W
+ REG_TLR = 0x07, // Trigger Level Register R/W
+ REG_TXLV = 0x08, // Transmitter FIFO Level register Read
+ REG_RXLV = 0x09, // Receiver FIFO Level register Read
+ REG_IODIR = 0x0A, // Programmable I/O pins Direction register R/W
+ REG_IOSTATE = 0x0B, // Programmable I/O pins State register R/W
+ REG_IOINTENA = 0x0C, // I/O Interrupt Enable register R/W
+ REG_IOCONTROL = 0x0E, // I/O Control register R/W
+ REG_EFCR = 0x0F, // Extra Features Control Register R/W
+ };
+
+// SC16IS752 Special register set
+enum : uint8_t{
+ REG_DLL = 0x00, // Division registers R/W
+ REG_DLH = 0x01, // Division registers R/W
+ };
+
+// SC16IS752 Enhanced regiter set
+enum : uint8_t{
+ REG_EFR = 0X02, // Enhanced Features Register R/W
+ REG_XON1 = 0x04, // R/W
+ REG_XON2 = 0x05, // R/W
+ REG_XOFF1 = 0x06, // R/W
+ REG_XOFF2 = 0x07, // R/W
+ };
+
diff --git a/IO_I2CRailcom.h b/IO_I2CRailcom.h
index 9dcccdf..2d1e519 100644
--- a/IO_I2CRailcom.h
+++ b/IO_I2CRailcom.h
@@ -45,10 +45,7 @@
#ifndef IO_I2CRailcom_h
#define IO_I2CRailcom_h
-#include "IODevice.h"
-#include "I2CManager.h"
-#include "DIAG.h"
-#include "DCCWaveform.h"
+#include "Arduino.h"
#include "Railcom.h"
// Debug and diagnostic defines, enable too many will result in slowing the driver
@@ -64,80 +61,13 @@ private:
Railcom * _channelMonitors[2];
public:
// Constructor
- I2CRailcom(VPIN firstVpin, int nPins, I2CAddress i2cAddress){
- _firstVpin = firstVpin;
- _nPins = nPins;
- _I2CAddress = i2cAddress;
- _channelMonitors[0]=new Railcom(firstVpin);
- if (nPins>1) _channelMonitors[1]=new Railcom(firstVpin+1);
- addDevice(this);
- }
+ I2CRailcom(VPIN firstVpin, int nPins, I2CAddress i2cAddress);
-public:
- static void create(VPIN firstVpin, int nPins, I2CAddress i2cAddress) {
- if (nPins>2) nPins=2;
- if (checkNoOverlap(firstVpin, nPins, i2cAddress))
- new I2CRailcom(firstVpin, nPins, i2cAddress);
- }
-
- void _begin() override {
- I2CManager.setClock(1000000); // TODO do we need this?
- I2CManager.begin();
- auto exists=I2CManager.exists(_I2CAddress);
- DIAG(F("I2CRailcom: %s UART%S detected"),
- _I2CAddress.toString(), exists?F(""):F(" NOT"));
- if (!exists) return;
+ static void create(VPIN firstVpin, int nPins, I2CAddress i2cAddress) ;
- _UART_CH=0;
- Init_SC16IS752(); // Initialize UART0
- if (_nPins>1) {
- _UART_CH=1;
- Init_SC16IS752(); // Initialize UART1
- }
-
- if (_deviceState==DEVSTATE_INITIALISING) _deviceState=DEVSTATE_NORMAL;
- _display();
- }
-
-
- void _loop(unsigned long currentMicros) override {
- // Read responses from device
- if (_deviceState!=DEVSTATE_NORMAL) return;
-
- // return if in cutout or cutout very soon.
- if (!DCCWaveform::isRailcomSampleWindow()) return;
-
- // IF we have 2 channels, flip channels each loop
- if (_nPins>1) _UART_CH=_UART_CH?0:1;
-
- // have we read this cutout already?
- auto cut=DCCWaveform::getRailcomCutoutCounter();
- if (cutoutCounter[_UART_CH]==cut) return;
- cutoutCounter[_UART_CH]=cut;
-
- // Read incoming raw Railcom data, and process accordingly
-
- auto inlength = UART_ReadRegister(REG_RXLV);
-
- if (inlength> sizeof(_inbuf)) inlength=sizeof(_inbuf);
- _inbuf[0]=0;
- if (inlength>0) {
- // Read data buffer from UART
- _outbuf[0]=(byte)(REG_RHR << 3 | _UART_CH << 1);
- I2CManager.read(_I2CAddress, _inbuf, inlength, _outbuf, 1);
- }
- // HK: Reset FIFO at end of read cycle
- UART_WriteRegister(REG_FCR, 0x07,false);
-
- // Ask Railcom to interpret the raw data
- _channelMonitors[_UART_CH]->process(_inbuf,inlength);
- }
-
-
- void _display() override {
- DIAG(F("I2CRailcom: Configured on Vpins:%u-%u %S"), _firstVpin, _firstVpin+_nPins-1,
- (_deviceState!=DEVSTATE_NORMAL) ? F("OFFLINE") : F(""));
- }
+ void _begin() ;
+ void _loop(unsigned long currentMicros) override ;
+ void _display() override ;
private:
@@ -168,115 +98,9 @@ private:
static const unsigned long SC16IS752_XTAL_FREQ_RAILCOM = 16000000; // Baud rate for Railcom signal
static const uint16_t _divisor = (SC16IS752_XTAL_FREQ_RAILCOM / PRESCALER) / (BAUD_RATE * 16);
- void Init_SC16IS752(){
- if (_UART_CH==0) { // HK: Currently fixed on ch 0
- // only reset on channel 0}
- UART_WriteRegister(REG_IOCONTROL, 0x08,false); // UART Software reset
- //_deviceState=DEVSTATE_INITIALISING; // ignores error during reset which seems normal. // HK: this line is moved to below
- auto iocontrol_readback = UART_ReadRegister(REG_IOCONTROL);
- if (iocontrol_readback == 0x00){
- _deviceState=DEVSTATE_INITIALISING;
- DIAG(F("I2CRailcom: %s SRESET readback: 0x%x"),_I2CAddress.toString(), iocontrol_readback);
- } else {
- DIAG(F("I2CRailcom: %s SRESET: 0x%x"),_I2CAddress.toString(), iocontrol_readback);
- }
- }
- // HK:
- // You write 0x08 to the IOCONTROL register, setting bit 3 (SRESET), as per datasheet 8.18:
- // "Software Reset. A write to this bit will reset the device. Once the
- // device is reset this bit is automatically set to logic 0"
- // So you can not readback the val you have written as this has changed.
- // I've added an extra UART_ReadRegister(REG_IOCONTROL) and check if the return value is 0x00
- // then set _deviceState=DEVSTATE_INITIALISING;
-
-
- // HK: only do clear FIFO at end of Init_SC16IS752
- //UART_WriteRegister(REG_FCR, 0x07,false); // Reset FIFO, clear RX & TX FIFO (write only)
-
- UART_WriteRegister(REG_MCR, 0x00); // Set MCR to all 0, includes Clock divisor
-
- //UART_WriteRegister(REG_LCR, 0x80); // Divisor latch enabled
-
- UART_WriteRegister(REG_LCR, 0x80 | WORD_LEN | STOP_BIT | PARITY_ENA | PARITY_TYPE); // Divisor latch enabled and comm parameters set
- UART_WriteRegister(REG_DLL, (uint8_t)_divisor); // Write DLL
- UART_WriteRegister(REG_DLH, (uint8_t)(_divisor >> 8)); // Write DLH
- auto lcr_readback = UART_ReadRegister(REG_LCR);
- lcr_readback = lcr_readback & 0x7F;
- UART_WriteRegister(REG_LCR, lcr_readback); // Divisor latch disabled
-
- //UART_WriteRegister(REG_LCR, WORD_LEN | STOP_BIT | PARITY_ENA | PARITY_TYPE); // Divisor latch disabled
-
- UART_WriteRegister(REG_FCR, 0x07,false); // Reset FIFO, clear RX & TX FIFO (write only)
-
- #ifdef DIAG_I2CRailcom
- // HK: Test to see if internal loopback works and if REG_RXLV increment to at least 0x01
- // Set REG_MCR bit 4 to 1, Enable Loopback
- UART_WriteRegister(REG_MCR, 0x10);
- UART_WriteRegister(REG_THR, 0x88, false); // Send 0x88
- auto inlen = UART_ReadRegister(REG_RXLV);
- if (inlen == 0){
- DIAG(F("I2CRailcom: Loopback test: %s/%d failed"),_I2CAddress.toString(), _UART_CH);
- } else {
- DIAG(F("Railcom: Loopback test: %s/%d RX Fifo lvl: 0x%x"),_I2CAddress.toString(), _UART_CH, inlen);
- _outbuf[0]=(byte)(REG_RHR << 3 | _UART_CH << 1);
- I2CManager.read(_I2CAddress, _inbuf, inlen, _outbuf, 1);
- #ifdef DIAG_I2CRailcom_data
- DIAG(F("Railcom: Loopback test: %s/%d RX FIFO Data"), _I2CAddress.toString(), _UART_CH);
- for (int i = 0; i < inlen; i++){
- DIAG(F("Railcom: Loopback data [0x%x]: 0x%x"), i, _inbuf[i]);
- //DIAG(F("[0x%x]: 0x%x"), i, _inbuf[i]);
- }
- #endif
- }
- UART_WriteRegister(REG_MCR, 0x00); // Set REG_MCR back to 0x00
- #endif
-
- #ifdef DIAG_I2CRailcom
- // Sent some data to check if UART baudrate is set correctly, check with logic analyzer on TX pin
- UART_WriteRegister(REG_THR, 9, false);
- DIAG(F("I2CRailcom: UART %s/%d Test TX = 0x09"),_I2CAddress.toString(), _UART_CH);
- #endif
-
- if (_deviceState==DEVSTATE_INITIALISING) {
- DIAG(F("I2CRailcom: UART %d init complete"),_UART_CH);
- }
- // HK: final FIFO reset
- UART_WriteRegister(REG_FCR, 0x07,false); // Reset FIFO, clear RX & TX FIFO (write only)
-
- }
-
-
-
-
- void UART_WriteRegister(uint8_t reg, uint8_t val, bool readback=true){
- _outbuf[0] = (byte)( reg << 3 | _UART_CH << 1);
- _outbuf[1]=val;
- auto status=I2CManager.write(_I2CAddress, _outbuf, (uint8_t)2);
- if(status!=I2C_STATUS_OK) {
- DIAG(F("I2CRailcom: %s/%d write reg=0x%x,data=0x%x,I2Cstate=%d"),
- _I2CAddress.toString(), _UART_CH, reg, val, status);
- _deviceState=DEVSTATE_FAILED;
- }
- if (readback) { // Read it back to cross check
- auto readback=UART_ReadRegister(reg);
- if (readback!=val) {
- DIAG(F("I2CRailcom readback: %s/%d reg:0x%x write=0x%x read=0x%x"),_I2CAddress.toString(),_UART_CH,reg,val,readback);
- }
- }
- }
-
-
- uint8_t UART_ReadRegister(uint8_t reg){
- _outbuf[0] = (byte)(reg << 3 | _UART_CH << 1); // _outbuffer[0] has now UART_REG and UART_CH
- _inbuf[0]=0;
- auto status=I2CManager.read(_I2CAddress, _inbuf, 1, _outbuf, 1);
- if (status!=I2C_STATUS_OK) {
- DIAG(F("I2CRailcom read: %s/%d read reg=0x%x,I2Cstate=%d"),
- _I2CAddress.toString(), _UART_CH, reg, status);
- _deviceState=DEVSTATE_FAILED;
- }
- return _inbuf[0];
- }
+ void Init_SC16IS752();
+ void UART_WriteRegister(uint8_t reg, uint8_t val, bool readback=true);
+ uint8_t UART_ReadRegister(uint8_t reg);
// SC16IS752 General register set (from the datasheet)
enum : uint8_t {
diff --git a/IO_TM1638.cpp b/IO_TM1638.cpp
index 48d4602..7a22e6b 100644
--- a/IO_TM1638.cpp
+++ b/IO_TM1638.cpp
@@ -22,6 +22,7 @@
#include
#include "IODevice.h"
+#include "IO_TM1638.h"
#include "DIAG.h"
diff --git a/version.h b/version.h
index 49007e9..b15e3d2 100644
--- a/version.h
+++ b/version.h
@@ -3,7 +3,8 @@
#include "StringFormatter.h"
-#define VERSION "5.5.5"
+#define VERSION "5.5.6"
+// 5.5.6 - Fix ESP32 build bug caused by include reference loop
// 5.5.5 - Railcom implementation with IO_I2CRailcom driver
// - response analysis and block management.
// - POM read using Railcom.