mirror of
https://github.com/DCC-EX/CommandStation-EX.git
synced 2025-04-03 20:20:12 +02:00
Compare commits
24 Commits
v5.2.96-De
...
master
Author | SHA1 | Date | |
---|---|---|---|
|
4125e73318 | ||
|
911bbd63be | ||
|
393b0bbd16 | ||
|
d9bd1e75f2 | ||
|
d1daf41f12 | ||
|
6bfa7028c4 | ||
|
a5d1d04882 | ||
|
bd6e426499 | ||
|
09bae44cc0 | ||
|
9f3354c687 | ||
|
fb495985f4 | ||
|
f868604ca9 | ||
|
41168a9dd8 | ||
|
0154e7fd78 | ||
|
9054d8d9f5 | ||
|
865f75dda4 | ||
|
b40fa779a6 | ||
|
2115ada2a1 | ||
|
830de850a9 | ||
|
c28965c58d | ||
|
0476b9c1d8 | ||
|
ba9ca1ccad | ||
|
c389fe9d3b | ||
|
79c30ec516 |
@ -280,6 +280,9 @@ void CommandDistributor::broadcastPower() {
|
|||||||
state = '1';
|
state = '1';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (state != '2')
|
||||||
|
broadcastReply(COMMAND_TYPE, F("<p%c>\n"),state);
|
||||||
|
|
||||||
// additional info about MAIN, PROG and JOIN
|
// additional info about MAIN, PROG and JOIN
|
||||||
bool main=TrackManager::getMainPower()==POWERMODE::ON;
|
bool main=TrackManager::getMainPower()==POWERMODE::ON;
|
||||||
bool prog=TrackManager::getProgPower()==POWERMODE::ON;
|
bool prog=TrackManager::getProgPower()==POWERMODE::ON;
|
||||||
@ -288,7 +291,7 @@ void CommandDistributor::broadcastPower() {
|
|||||||
const FSH * reason=F("");
|
const FSH * reason=F("");
|
||||||
if (join) {
|
if (join) {
|
||||||
reason = F(" JOIN"); // with space at start so we can append without space
|
reason = F(" JOIN"); // with space at start so we can append without space
|
||||||
broadcastReply(COMMAND_TYPE, F("<p1 %S>\n"),reason);
|
broadcastReply(COMMAND_TYPE, F("<p1%S>\n"),reason);
|
||||||
} else {
|
} else {
|
||||||
if (main) {
|
if (main) {
|
||||||
//reason = F("MAIN");
|
//reason = F("MAIN");
|
||||||
@ -299,9 +302,6 @@ void CommandDistributor::broadcastPower() {
|
|||||||
broadcastReply(COMMAND_TYPE, F("<p1 PROG>\n"));
|
broadcastReply(COMMAND_TYPE, F("<p1 PROG>\n"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (state != '2')
|
|
||||||
broadcastReply(COMMAND_TYPE, F("<p%c>\n"),state);
|
|
||||||
#ifdef CD_HANDLE_RING
|
#ifdef CD_HANDLE_RING
|
||||||
// send '1' if all main are on, otherwise global state (which in that case is '0' or '2')
|
// send '1' if all main are on, otherwise global state (which in that case is '0' or '2')
|
||||||
broadcastReply(WITHROTTLE_TYPE, F("PPA%c\n"), main?'1': state);
|
broadcastReply(WITHROTTLE_TYPE, F("PPA%c\n"), main?'1': state);
|
||||||
|
12
DCC.cpp
12
DCC.cpp
@ -229,15 +229,9 @@ bool DCC::setFn( int cab, int16_t functionNumber, bool on) {
|
|||||||
|
|
||||||
// Flip function state (used from withrottle protocol)
|
// Flip function state (used from withrottle protocol)
|
||||||
void DCC::changeFn( int cab, int16_t functionNumber) {
|
void DCC::changeFn( int cab, int16_t functionNumber) {
|
||||||
if (cab<=0 || functionNumber>31) return;
|
auto currentValue=getFn(cab,functionNumber);
|
||||||
int reg = lookupSpeedTable(cab);
|
if (currentValue<0) return; // function not valid for change
|
||||||
if (reg<0) return;
|
setFn(cab,functionNumber, currentValue?false:true);
|
||||||
unsigned long funcmask = (1UL<<functionNumber);
|
|
||||||
speedTable[reg].functions ^= funcmask;
|
|
||||||
if (functionNumber <= 28) {
|
|
||||||
updateGroupflags(speedTable[reg].groupFlags, functionNumber);
|
|
||||||
}
|
|
||||||
CommandDistributor::broadcastLoco(reg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Report function state (used from withrottle protocol)
|
// Report function state (used from withrottle protocol)
|
||||||
|
@ -167,8 +167,10 @@ int16_t DCCEXParser::splitValues(int16_t result[MAX_COMMAND_PARAMS], byte *cmd,
|
|||||||
break;
|
break;
|
||||||
if (hot == '\0')
|
if (hot == '\0')
|
||||||
return -1;
|
return -1;
|
||||||
if (hot == '>')
|
if (hot == '>') {
|
||||||
|
*remainingCmd = '\0'; // terminate the cmd string with 0 instead of '>'
|
||||||
return parameterCount;
|
return parameterCount;
|
||||||
|
}
|
||||||
state = 2;
|
state = 2;
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@ -265,17 +267,22 @@ void DCCEXParser::parse(const FSH * cmd) {
|
|||||||
// See documentation on DCC class for info on this section
|
// See documentation on DCC class for info on this section
|
||||||
|
|
||||||
void DCCEXParser::parse(Print *stream, byte *com, RingStream *ringStream) {
|
void DCCEXParser::parse(Print *stream, byte *com, RingStream *ringStream) {
|
||||||
// This function can get stings of the form "<C OMM AND>" or "C OMM AND"
|
// This function can get stings of the form "<C OMM AND>" or "C OMM AND>"
|
||||||
// found is true first after the leading "<" has been passed
|
// found is true first after the leading "<" has been passed which results
|
||||||
|
// in parseOne() getting c="C OMM AND>"
|
||||||
|
byte *cForLater = NULL;
|
||||||
bool found = (com[0] != '<');
|
bool found = (com[0] != '<');
|
||||||
for (byte *c=com; c[0] != '\0'; c++) {
|
for (byte *c=com; c[0] != '\0'; c++) {
|
||||||
if (found) {
|
if (found) {
|
||||||
parseOne(stream, c, ringStream);
|
cForLater = c;
|
||||||
found=false;
|
found=false;
|
||||||
}
|
}
|
||||||
if (c[0] == '<')
|
if (c[0] == '<') {
|
||||||
|
if (cForLater) parseOne(stream, cForLater, ringStream);
|
||||||
found = true;
|
found = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
if (cForLater) parseOne(stream, cForLater, ringStream);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DCCEXParser::parseOne(Print *stream, byte *com, RingStream * ringStream)
|
void DCCEXParser::parseOne(Print *stream, byte *com, RingStream * ringStream)
|
||||||
@ -403,7 +410,7 @@ void DCCEXParser::parseOne(Print *stream, byte *com, RingStream * ringStream)
|
|||||||
) break;
|
) break;
|
||||||
// Honour the configuration option (config.h) which allows the <a> command to be reversed
|
// Honour the configuration option (config.h) which allows the <a> command to be reversed
|
||||||
// Because of earlier confusion we need to do the same thing under both defines
|
// Because of earlier confusion we need to do the same thing under both defines
|
||||||
#if defined(DCC_ACCESSORY_COMMAND_REVERSE) || defined(DCC_ACCESSORY_RCN_213)
|
#if defined(DCC_ACCESSORY_COMMAND_REVERSE)
|
||||||
DCC::setAccessory(address, subaddress,p[activep]==0,onoff);
|
DCC::setAccessory(address, subaddress,p[activep]==0,onoff);
|
||||||
#else
|
#else
|
||||||
DCC::setAccessory(address, subaddress,p[activep]==1,onoff);
|
DCC::setAccessory(address, subaddress,p[activep]==1,onoff);
|
||||||
|
6
DCCRMT.h
6
DCCRMT.h
@ -44,6 +44,12 @@ class RMTChannel {
|
|||||||
return true;
|
return true;
|
||||||
return dataReady;
|
return dataReady;
|
||||||
};
|
};
|
||||||
|
inline void waitForDataCopy() {
|
||||||
|
while(1) { // do nothing and wait for interrupt clearing dataReady to happen
|
||||||
|
if (dataReady == false)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
inline uint32_t packetCount() { return packetCounter; };
|
inline uint32_t packetCount() { return packetCounter; };
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -78,11 +78,17 @@ int DCCTimer::freeMemory() {
|
|||||||
////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////
|
||||||
#ifdef ARDUINO_ARCH_ESP32
|
#ifdef ARDUINO_ARCH_ESP32
|
||||||
|
|
||||||
|
#if __has_include("esp_idf_version.h")
|
||||||
#include "esp_idf_version.h"
|
#include "esp_idf_version.h"
|
||||||
#if ESP_IDF_VERSION_MAJOR > 4
|
#endif
|
||||||
|
#if ESP_IDF_VERSION_MAJOR == 4
|
||||||
|
// all well correct IDF version
|
||||||
|
#else
|
||||||
#error "DCC-EX does not support compiling with IDF version 5.0 or later. Downgrade your ESP32 library to a version that contains IDF version 4. Arduino ESP32 library 3.0.0 is too new. Downgrade to one of 2.0.9 to 2.0.17"
|
#error "DCC-EX does not support compiling with IDF version 5.0 or later. Downgrade your ESP32 library to a version that contains IDF version 4. Arduino ESP32 library 3.0.0 is too new. Downgrade to one of 2.0.9 to 2.0.17"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// protect all the rest of the code from IDF version 5
|
||||||
|
#if ESP_IDF_VERSION_MAJOR == 4
|
||||||
#include "DIAG.h"
|
#include "DIAG.h"
|
||||||
#include <driver/adc.h>
|
#include <driver/adc.h>
|
||||||
#include <soc/sens_reg.h>
|
#include <soc/sens_reg.h>
|
||||||
@ -322,5 +328,5 @@ void ADCee::scan() {
|
|||||||
|
|
||||||
void ADCee::begin() {
|
void ADCee::begin() {
|
||||||
}
|
}
|
||||||
|
#endif //IDF v4
|
||||||
#endif //ESP32
|
#endif //ESP32
|
||||||
|
@ -278,7 +278,11 @@ void DCCWaveform::begin() {
|
|||||||
|
|
||||||
void DCCWaveform::schedulePacket(const byte buffer[], byte byteCount, byte repeats) {
|
void DCCWaveform::schedulePacket(const byte buffer[], byte byteCount, byte repeats) {
|
||||||
if (byteCount > MAX_PACKET_SIZE) return; // allow for chksum
|
if (byteCount > MAX_PACKET_SIZE) return; // allow for chksum
|
||||||
|
RMTChannel *rmtchannel = (isMainTrack ? rmtMainChannel : rmtProgChannel);
|
||||||
|
if (rmtchannel == NULL)
|
||||||
|
return; // no idea to prepare packet if we can not send it anyway
|
||||||
|
|
||||||
|
rmtchannel->waitForDataCopy(); // blocking wait so we can write into buffer
|
||||||
byte checksum = 0;
|
byte checksum = 0;
|
||||||
for (byte b = 0; b < byteCount; b++) {
|
for (byte b = 0; b < byteCount; b++) {
|
||||||
checksum ^= buffer[b];
|
checksum ^= buffer[b];
|
||||||
@ -296,13 +300,7 @@ void DCCWaveform::schedulePacket(const byte buffer[], byte byteCount, byte repea
|
|||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
do {
|
do {
|
||||||
if(isMainTrack) {
|
ret = rmtchannel->RMTfillData(pendingPacket, pendingLength, pendingRepeats);
|
||||||
if (rmtMainChannel != NULL)
|
|
||||||
ret = rmtMainChannel->RMTfillData(pendingPacket, pendingLength, pendingRepeats);
|
|
||||||
} else {
|
|
||||||
if (rmtProgChannel != NULL)
|
|
||||||
ret = rmtProgChannel->RMTfillData(pendingPacket, pendingLength, pendingRepeats);
|
|
||||||
}
|
|
||||||
} while(ret > 0);
|
} while(ret > 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -931,8 +931,9 @@ void RMFT2::loop2() {
|
|||||||
|
|
||||||
#ifndef DISABLE_PROG
|
#ifndef DISABLE_PROG
|
||||||
case OPCODE_JOIN:
|
case OPCODE_JOIN:
|
||||||
TrackManager::setPower(POWERMODE::ON);
|
|
||||||
TrackManager::setJoin(true);
|
TrackManager::setJoin(true);
|
||||||
|
TrackManager::setMainPower(POWERMODE::ON);
|
||||||
|
TrackManager::setProgPower(POWERMODE::ON);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case OPCODE_UNJOIN:
|
case OPCODE_UNJOIN:
|
||||||
|
@ -217,7 +217,7 @@
|
|||||||
#define BROADCAST(msg)
|
#define BROADCAST(msg)
|
||||||
#define CALL(route)
|
#define CALL(route)
|
||||||
#define CLEAR_STASH(id)
|
#define CLEAR_STASH(id)
|
||||||
#define CLEAR_ALL_STASH(id)
|
#define CLEAR_ALL_STASH
|
||||||
#define CLOSE(id)
|
#define CLOSE(id)
|
||||||
#define CONFIGURE_SERVO(vpin,pos1,pos2,profile)
|
#define CONFIGURE_SERVO(vpin,pos1,pos2,profile)
|
||||||
#define DCC_SIGNAL(id,add,subaddr)
|
#define DCC_SIGNAL(id,add,subaddr)
|
||||||
|
@ -1 +1 @@
|
|||||||
#define GITHUB_SHA "devel-202501091941Z"
|
#define GITHUB_SHA "c389fe9"
|
||||||
|
BIN
Release_Notes/DCC-EX v5.4 Release Notes.xlsx
Normal file
BIN
Release_Notes/DCC-EX v5.4 Release Notes.xlsx
Normal file
Binary file not shown.
@ -126,29 +126,33 @@ void SerialManager::loop2() {
|
|||||||
buffer[0] = '\0';
|
buffer[0] = '\0';
|
||||||
}
|
}
|
||||||
} else { // if (inCommandPayload)
|
} else { // if (inCommandPayload)
|
||||||
if (bufferLength < (COMMAND_BUFFER_SIZE-1))
|
if (bufferLength < (COMMAND_BUFFER_SIZE-1)) {
|
||||||
buffer[bufferLength++] = ch;
|
buffer[bufferLength++] = ch; // advance bufferLength
|
||||||
if (inCommandPayload > PAYLOAD_NORMAL) {
|
if (inCommandPayload > PAYLOAD_NORMAL) {
|
||||||
if (inCommandPayload > 32 + 2) { // String way too long
|
if (inCommandPayload > 32 + 2) { // String way too long
|
||||||
ch = '>'; // we end this nonsense
|
ch = '>'; // we end this nonsense
|
||||||
inCommandPayload = PAYLOAD_NORMAL;
|
inCommandPayload = PAYLOAD_NORMAL;
|
||||||
DIAG(F("Parse error: Unbalanced string"));
|
DIAG(F("Parse error: Unbalanced string"));
|
||||||
// fall through to ending parsing below
|
// fall through to ending parsing below
|
||||||
} else if (ch == '"') { // String end
|
} else if (ch == '"') { // String end
|
||||||
inCommandPayload = PAYLOAD_NORMAL;
|
inCommandPayload = PAYLOAD_NORMAL;
|
||||||
continue; // do not fall through
|
continue; // do not fall through
|
||||||
} else
|
} else
|
||||||
inCommandPayload++;
|
inCommandPayload++;
|
||||||
}
|
}
|
||||||
if (inCommandPayload == PAYLOAD_NORMAL) {
|
if (inCommandPayload == PAYLOAD_NORMAL) {
|
||||||
if (ch == '>') {
|
if (ch == '>') {
|
||||||
buffer[bufferLength] = '\0';
|
buffer[bufferLength] = '\0'; // This \0 is after the '>'
|
||||||
DCCEXParser::parse(serial, buffer, NULL);
|
DCCEXParser::parse(serial, buffer, NULL); // buffer parsed with trailing '>'
|
||||||
inCommandPayload = PAYLOAD_FALSE;
|
inCommandPayload = PAYLOAD_FALSE;
|
||||||
break;
|
break;
|
||||||
} else if (ch == '"') {
|
} else if (ch == '"') {
|
||||||
inCommandPayload = PAYLOAD_STRING;
|
inCommandPayload = PAYLOAD_STRING;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
DIAG(F("Parse error: input buffer overflow"));
|
||||||
|
inCommandPayload = PAYLOAD_FALSE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -668,7 +668,8 @@ void TrackManager::setJoin(bool joined) {
|
|||||||
if (track[t]->getMode() & TRACK_MODE_PROG) { // find PROG track
|
if (track[t]->getMode() & TRACK_MODE_PROG) { // find PROG track
|
||||||
tempProgTrack = t; // remember PROG track
|
tempProgTrack = t; // remember PROG track
|
||||||
setTrackMode(t, TRACK_MODE_MAIN);
|
setTrackMode(t, TRACK_MODE_MAIN);
|
||||||
track[t]->setPower(POWERMODE::ON); // if joined, always on
|
// setPower() of the track called after
|
||||||
|
// seperately after setJoin() instead
|
||||||
break; // there is only one prog track, done
|
break; // there is only one prog track, done
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -680,8 +681,6 @@ void TrackManager::setJoin(bool joined) {
|
|||||||
setTrackMode(tempProgTrack, TRACK_MODE_PROG); // set track mode back to prog
|
setTrackMode(tempProgTrack, TRACK_MODE_PROG); // set track mode back to prog
|
||||||
track[tempProgTrack]->setPower(tPTmode); // set power status as it was before
|
track[tempProgTrack]->setPower(tPTmode); // set power status as it was before
|
||||||
tempProgTrack = MAX_TRACKS+1;
|
tempProgTrack = MAX_TRACKS+1;
|
||||||
} else {
|
|
||||||
DIAG(F("Unjoin but no remembered prog track"));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -379,7 +379,7 @@
|
|||||||
// DCC++ Classic behaviour is that Throw writes a 1 in the packet,
|
// DCC++ Classic behaviour is that Throw writes a 1 in the packet,
|
||||||
// and Close writes a 0.
|
// and Close writes a 0.
|
||||||
// RCN-213 specifies that Throw is 0 and Close is 1.
|
// RCN-213 specifies that Throw is 0 and Close is 1.
|
||||||
#if defined(DCC_TURNOUTS_RCN_213)
|
#ifndef DCC_TURNOUTS_RCN_213
|
||||||
close = !close;
|
close = !close;
|
||||||
#endif
|
#endif
|
||||||
DCC::setAccessory(_dccTurnoutData.address, _dccTurnoutData.subAddress, close);
|
DCC::setAccessory(_dccTurnoutData.address, _dccTurnoutData.subAddress, close);
|
||||||
|
@ -500,9 +500,9 @@ void WiThrottle::getLocoCallback(int16_t locoid) {
|
|||||||
char addcmd[20]={'M',stashThrottleChar,'+', addrchar};
|
char addcmd[20]={'M',stashThrottleChar,'+', addrchar};
|
||||||
itoa(locoid,addcmd+4,10);
|
itoa(locoid,addcmd+4,10);
|
||||||
stashInstance->multithrottle(stashStream, (byte *)addcmd);
|
stashInstance->multithrottle(stashStream, (byte *)addcmd);
|
||||||
|
TrackManager::setJoin(true); // <1 JOIN> so we can drive loco away
|
||||||
TrackManager::setMainPower(POWERMODE::ON);
|
TrackManager::setMainPower(POWERMODE::ON);
|
||||||
TrackManager::setProgPower(POWERMODE::ON);
|
TrackManager::setProgPower(POWERMODE::ON);
|
||||||
TrackManager::setJoin(true); // <1 JOIN> so we can drive loco away
|
|
||||||
DIAG(F("LocoCallback commit success"));
|
DIAG(F("LocoCallback commit success"));
|
||||||
stashStream->commit();
|
stashStream->commit();
|
||||||
}
|
}
|
||||||
|
@ -269,8 +269,8 @@ The configuration file for DCC-EX Command Station
|
|||||||
// over DCC++. This #define likewise inverts the behaviour of the <a> command
|
// over DCC++. This #define likewise inverts the behaviour of the <a> command
|
||||||
// for triggering DCC Accessory Decoders, so that <a addr subaddr 0> generates a
|
// for triggering DCC Accessory Decoders, so that <a addr subaddr 0> generates a
|
||||||
// DCC packet with D=1 (close turnout) and <a addr subaddr 1> generates D=0
|
// DCC packet with D=1 (close turnout) and <a addr subaddr 1> generates D=0
|
||||||
// (throw turnout). This is the same as DCC_ACCESSORY_COMMAND_REVERSE
|
// (throw turnout).
|
||||||
//#define DCC_ACCESSORY_RCN_213
|
//#define DCC_ACCESSORY_COMMAND_REVERSE
|
||||||
|
|
||||||
|
|
||||||
// HANDLING MULTIPLE SERIAL THROTTLES
|
// HANDLING MULTIPLE SERIAL THROTTLES
|
||||||
|
@ -12,18 +12,10 @@
|
|||||||
default_envs =
|
default_envs =
|
||||||
mega2560
|
mega2560
|
||||||
uno
|
uno
|
||||||
unowifiR2
|
|
||||||
nano
|
nano
|
||||||
samd21-dev-usb
|
|
||||||
samd21-zero-usb
|
|
||||||
ESP32
|
ESP32
|
||||||
Nucleo-F411RE
|
Nucleo-F411RE
|
||||||
Nucleo-F446RE
|
Nucleo-F446RE
|
||||||
Teensy3_2
|
|
||||||
Teensy3_5
|
|
||||||
Teensy3_6
|
|
||||||
Teensy4_0
|
|
||||||
Teensy4_1
|
|
||||||
src_dir = .
|
src_dir = .
|
||||||
include_dir = .
|
include_dir = .
|
||||||
|
|
||||||
|
11
version.h
11
version.h
@ -3,7 +3,16 @@
|
|||||||
|
|
||||||
#include "StringFormatter.h"
|
#include "StringFormatter.h"
|
||||||
|
|
||||||
#define VERSION "5.2.96"
|
#define VERSION "5.4.7"
|
||||||
|
// 5.4.7 - Bugfix: EXRAIL fix CLEAR_ALL_STASH
|
||||||
|
// 5.4.6 - Bugfix: Do not drop further commands in same packet
|
||||||
|
// 5.4.5 - ESP32: Better detection of correct IDF version
|
||||||
|
// - track power is always turned on after setJoin() not by setJoin()
|
||||||
|
// 5.4.4 - bugfix in parser, input buffer overrun and trailing > that did break <+>
|
||||||
|
// 5.4.3 - bugfix changeFn for functions 29..31
|
||||||
|
// 5.4.2 - Reversed turnout bugfix
|
||||||
|
// 5.4.1 - ESP32 bugfix packet buffer race
|
||||||
|
// 5.4.0 - New version on master
|
||||||
// 5.2.96 - EXRAIL additions XFWD() and XREV()
|
// 5.2.96 - EXRAIL additions XFWD() and XREV()
|
||||||
// 5.2.95 - Release candidate for 5.4
|
// 5.2.95 - Release candidate for 5.4
|
||||||
// 5.2.94 - Bugfix: Less confusion and simpler code around the RCN213 defines
|
// 5.2.94 - Bugfix: Less confusion and simpler code around the RCN213 defines
|
||||||
|
Loading…
Reference in New Issue
Block a user