mirror of
https://github.com/DCC-EX/CommandStation-EX.git
synced 2025-04-04 12:40:12 +02:00
829 lines
32 KiB
C++
829 lines
32 KiB
C++
/*
|
|
* © 2022 Paul M Antoine
|
|
* © 2021 Neil McKechnie
|
|
* © 2021 Mike S
|
|
* © 2021-2025 Herb Morton
|
|
* © 2020-2023 Harald Barth
|
|
* © 2020-2021 M Steve Todd
|
|
* © 2020-2021 Fred Decker
|
|
* © 2020-2025 Chris Harlow
|
|
* © 2022 Colin Murdoch
|
|
* All rights reserved.
|
|
*
|
|
* This file is part of CommandStation-EX
|
|
*
|
|
* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/*
|
|
List of single character OPCODEs in use for reference.
|
|
|
|
When determining a new OPCODE for a new feature, refer to this list as the source of truth.
|
|
|
|
Once a new OPCODE is decided upon, update this list.
|
|
|
|
Character, Usage
|
|
/, |EX-R| interactive commands
|
|
-, Remove from reminder table
|
|
=, |TM| configuration
|
|
!, Emergency stop
|
|
@, Reserved for future use - LCD messages to JMRI
|
|
#, Request number of supported cabs/locos; heartbeat
|
|
+, WiFi AT commands
|
|
?, Reserved for future use
|
|
0, Track power off
|
|
1, Track power on
|
|
a, DCC accessory control
|
|
A, DCC extended accessory control
|
|
b, Write CV bit on main
|
|
B, Write CV bit
|
|
c, Request current command
|
|
C, configure the CS
|
|
d,
|
|
D, Diagnostic commands
|
|
e, Erase EEPROM
|
|
E, Store configuration in EEPROM
|
|
f, Loco decoder function control (deprecated)
|
|
F, Loco decoder function control
|
|
g,
|
|
G,
|
|
h,
|
|
H, Turnout state broadcast
|
|
i, Server details string
|
|
I, Turntable object command, control, and broadcast
|
|
j, Throttle responses
|
|
J, Throttle queries
|
|
k, Block exit (Railcom)
|
|
K, Block enter (Railcom)
|
|
l, Loco speedbyte/function map broadcast
|
|
L, Reserved for LCC interface (implemented in EXRAIL)
|
|
m, message to throttles (broadcast output)
|
|
m, set momentum
|
|
M, Write DCC packet
|
|
n, Reserved for SensorCam
|
|
N, Reserved for Sensorcam
|
|
o, Neopixel driver (see also IO_NeoPixel.h)
|
|
O, Output broadcast
|
|
p, Broadcast power state
|
|
P, Write DCC packet
|
|
q, Sensor deactivated
|
|
Q, Sensor activated
|
|
r, Broadcast address read on programming track
|
|
R, Read CVs
|
|
s, Display status
|
|
S, Sensor configuration
|
|
t, Cab/loco update command
|
|
T, Turnout configuration/control
|
|
u, Reserved for user commands
|
|
U, Reserved for user commands
|
|
v,
|
|
V, Verify CVs
|
|
w, Write CV on main
|
|
W, Write CV
|
|
x,
|
|
X, Invalid command response
|
|
y,
|
|
Y, Output broadcast
|
|
z, Direct output
|
|
Z, Output configuration/control
|
|
*/
|
|
|
|
#include "StringFormatter.h"
|
|
#include "DCCEXParser.h"
|
|
#include "DCC.h"
|
|
#include "DCCWaveform.h"
|
|
#include "Turnouts.h"
|
|
#include "Outputs.h"
|
|
#include "Sensors.h"
|
|
#include "GITHUB_SHA.h"
|
|
#include "version.h"
|
|
#include "defines.h"
|
|
#include "CommandDistributor.h"
|
|
#include "EEStore.h"
|
|
#include "DIAG.h"
|
|
#include "TrackManager.h"
|
|
#include "DCCTimer.h"
|
|
#include "EXRAIL2.h"
|
|
#include "Turntables.h"
|
|
#include "version.h"
|
|
#include "KeywordHasher.h"
|
|
#include "CamParser.h"
|
|
#include "Stash.h"
|
|
#include "DCCEXParserMacros.h"
|
|
|
|
#ifdef ARDUINO_ARCH_ESP32
|
|
#include "WifiESP32.h"
|
|
#endif
|
|
|
|
// This macro can't be created easily as a portable function because the
|
|
// flashlist requires a far pointer for high flash access.
|
|
#define SENDFLASHLIST(stream,flashList) \
|
|
for (int16_t i=0;;i+=sizeof(flashList[0])) { \
|
|
int16_t value=GETHIGHFLASHW(flashList,i); \
|
|
if (value==INT16_MAX) break; \
|
|
StringFormatter::send(stream,F(" %d"),value); \
|
|
}
|
|
|
|
int16_t DCCEXParser::stashP[MAX_COMMAND_PARAMS];
|
|
bool DCCEXParser::stashBusy;
|
|
Print *DCCEXParser::stashStream = NULL;
|
|
RingStream *DCCEXParser::stashRingStream = NULL;
|
|
byte DCCEXParser::stashTarget=0;
|
|
|
|
// This is a JMRI command parser.
|
|
// It doesnt know how the string got here, nor how it gets back.
|
|
// It knows nothing about hardware or tracks... it just parses strings and
|
|
// calls the corresponding DCC api.
|
|
// Non-DCC things like turnouts, pins and sensors are handled in additional JMRI interface classes.
|
|
|
|
|
|
int16_t DCCEXParser::splitValues(int16_t result[MAX_COMMAND_PARAMS], byte *cmd, bool usehex)
|
|
{
|
|
byte state = 1;
|
|
byte parameterCount = 0;
|
|
int16_t runningValue = 0;
|
|
byte *remainingCmd = cmd + 1; // skips the opcode
|
|
bool signNegative = false;
|
|
|
|
// clear all parameters in case not enough found
|
|
for (int16_t i = 0; i < MAX_COMMAND_PARAMS; i++)
|
|
result[i] = 0;
|
|
|
|
while (parameterCount < MAX_COMMAND_PARAMS)
|
|
{
|
|
byte hot = *remainingCmd;
|
|
switch (state)
|
|
{
|
|
|
|
case 1: // skipping spaces before a param
|
|
if (hot == ' ')
|
|
break;
|
|
if (hot == '\0')
|
|
return -1;
|
|
if (hot == '>')
|
|
return parameterCount;
|
|
state = 2;
|
|
continue;
|
|
|
|
case 2: // checking sign or quoted string
|
|
#ifdef HAS_ENOUGH_MEMORY
|
|
if (hot == '"') {
|
|
// this inserts an extra parameter 0x7777 in front
|
|
// of each string parameter as a marker that can
|
|
// be checked that a string parameter follows
|
|
// This clashes of course with the real value
|
|
// 0x7777 which we hope is used seldom
|
|
result[parameterCount] = (int16_t)0x7777;
|
|
parameterCount++;
|
|
result[parameterCount] = (int16_t)(remainingCmd - cmd + 1);
|
|
parameterCount++;
|
|
state = 4;
|
|
break;
|
|
}
|
|
#endif
|
|
signNegative = false;
|
|
runningValue = 0;
|
|
state = 3;
|
|
if (hot != '-')
|
|
continue;
|
|
signNegative = true;
|
|
break;
|
|
case 3: // building a parameter
|
|
if (hot >= '0' && hot <= '9')
|
|
{
|
|
runningValue = (usehex?16:10) * runningValue + (hot - '0');
|
|
break;
|
|
}
|
|
if (hot >= 'a' && hot <= 'z') hot=hot-'a'+'A'; // uppercase a..z
|
|
if (usehex && hot>='A' && hot<='F') {
|
|
// treat A..F as hex not keyword
|
|
runningValue = 16 * runningValue + (hot - 'A' + 10);
|
|
break;
|
|
}
|
|
if (hot=='_' || (hot >= 'A' && hot <= 'Z'))
|
|
{
|
|
// Since JMRI got modified to send keywords in some rare cases, we need this
|
|
// Super Kluge to turn keywords into a hash value that can be recognised later
|
|
runningValue = ((runningValue << 5) + runningValue) ^ hot;
|
|
break;
|
|
}
|
|
result[parameterCount] = runningValue * (signNegative ? -1 : 1);
|
|
parameterCount++;
|
|
state = 1;
|
|
continue;
|
|
#ifdef HAS_ENOUGH_MEMORY
|
|
case 4: // skipover text
|
|
if (hot == '\0') // We did run to end of buffer without finding the "
|
|
return -1;
|
|
if (hot == '"') {
|
|
*remainingCmd = '\0'; // overwrite " in command buffer with the end-of-string
|
|
state = 1;
|
|
}
|
|
break;
|
|
#endif
|
|
}
|
|
remainingCmd++;
|
|
}
|
|
return parameterCount;
|
|
}
|
|
|
|
extern __attribute__((weak)) void myFilter(Print * stream, byte & opcode, byte & paramCount, int16_t p[]);
|
|
FILTER_CALLBACK DCCEXParser::filterCallback = myFilter;
|
|
FILTER_CALLBACK DCCEXParser::filterRMFTCallback = 0;
|
|
FILTER_CALLBACK DCCEXParser::filterCamParserCallback = 0;
|
|
AT_COMMAND_CALLBACK DCCEXParser::atCommandCallback = 0;
|
|
|
|
// deprecated
|
|
void DCCEXParser::setFilter(FILTER_CALLBACK filter)
|
|
{
|
|
filterCallback = filter;
|
|
}
|
|
void DCCEXParser::setRMFTFilter(FILTER_CALLBACK filter)
|
|
{
|
|
filterRMFTCallback = filter;
|
|
}
|
|
void DCCEXParser::setCamParserFilter(FILTER_CALLBACK filter)
|
|
{
|
|
filterCamParserCallback = filter;
|
|
}
|
|
void DCCEXParser::setAtCommandCallback(AT_COMMAND_CALLBACK callback)
|
|
{
|
|
atCommandCallback = callback;
|
|
}
|
|
|
|
// Parse an F() string
|
|
void DCCEXParser::parse(const FSH * cmd) {
|
|
DIAG(F("SETUP(\"%S\")"),cmd);
|
|
int size=STRLEN_P((char *)cmd)+1;
|
|
char buffer[size];
|
|
STRCPY_P(buffer,(char *)cmd);
|
|
parse(&USB_SERIAL,(byte *)buffer,NULL);
|
|
}
|
|
|
|
// See documentation on DCC class for info on this section
|
|
|
|
void DCCEXParser::parse(Print *stream, byte *com, RingStream *ringStream) {
|
|
// 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
|
|
bool found = (com[0] != '<');
|
|
for (byte *c=com; c[0] != '\0'; c++) {
|
|
if (found) {
|
|
parseOne(stream, c, ringStream);
|
|
found=false;
|
|
}
|
|
if (c[0] == '<')
|
|
found = true;
|
|
}
|
|
}
|
|
|
|
void DCCEXParser::parseOne(Print *stream, byte *com, RingStream * ringStream)
|
|
{
|
|
#ifdef DISABLE_PROG
|
|
(void)ringStream;
|
|
#endif
|
|
#ifndef DISABLE_EEPROM
|
|
(void)EEPROM; // tell compiler not to warn this is unused
|
|
#endif
|
|
byte params = 0;
|
|
if (Diag::CMD)
|
|
DIAG(F("PARSING:%s"), com);
|
|
int16_t p[MAX_COMMAND_PARAMS];
|
|
while (com[0] == '<' || com[0] == ' ')
|
|
com++; // strip off any number of < or spaces
|
|
byte opcode = com[0];
|
|
|
|
if (opcode=='+') {
|
|
if (atCommandCallback && !ringStream) {
|
|
TrackManager::setPower(POWERMODE::OFF);
|
|
atCommandCallback((HardwareSerial *)stream,com);
|
|
}
|
|
return; // we cant parse the <+ wifistuff > here
|
|
}
|
|
|
|
int16_t splitnum = splitValues(p, com, opcode=='M' || opcode=='P');
|
|
if (splitnum<0 || splitnum>=MAX_COMMAND_PARAMS) {
|
|
DIAG(F("Too many parameters"));
|
|
return;
|
|
}
|
|
|
|
// Because of check above we are now inside byte size
|
|
params = splitnum;
|
|
|
|
if (filterCallback)
|
|
filterCallback(stream, opcode, params, p);
|
|
if (filterRMFTCallback && opcode!='\0')
|
|
filterRMFTCallback(stream, opcode, params, p);
|
|
if (filterCamParserCallback && opcode!='\0')
|
|
filterCamParserCallback(stream, opcode, params, p);
|
|
if (opcode=='\0') return; // filterCallback asked us to ignore
|
|
|
|
// todo, revamp as filter
|
|
if (opcode=='=') // TRACK MANAGER CONTROL <= [params]>
|
|
if (TrackManager::parseEqualSign(stream, params, p))
|
|
return;
|
|
|
|
|
|
if (execute(com,stream, opcode, params, p, ringStream)) return;
|
|
|
|
// TODO magnificent diagnostics
|
|
StringFormatter::send(stream, F("<X>\n"), opcode);
|
|
if (opcode >= ' ' && opcode <= '~') {
|
|
DIAG(F("Opcode=%c params=%d"), opcode, params);
|
|
for (int i = 0; i < params; i++)
|
|
DIAG(F("p[%d]=%d (0x%x)"), i, p[i], p[i]);
|
|
} else {
|
|
DIAG(F("Unprintable %x"), opcode);
|
|
}
|
|
|
|
}
|
|
|
|
bool DCCEXParser::setThrottle(int16_t cab,int16_t tspeed,int16_t direction) {
|
|
// Convert DCC-EX protocol speed steps where
|
|
// -1=emergency stop, 0-126 as speeds
|
|
// to DCC 0=stop, 1= emergency stop, 2-127 speeds
|
|
if (tspeed > 126 || tspeed < -1) return false; // invalid JMRI speed code
|
|
if (tspeed < 0) tspeed = 1; // emergency stop DCC speed
|
|
else if (tspeed > 0) tspeed++; // map 1-126 -> 2-127
|
|
if (cab == 0 && tspeed > 1) return false; // ignore broadcasts of speed>1
|
|
if (direction < 0 || direction > 1) return false; // invalid direction code
|
|
if (cab > 10239 || cab < 0) return false; // beyond DCC range
|
|
|
|
DCC::setThrottle(cab, tspeed, direction);
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//===================================
|
|
|
|
bool DCCEXParser::funcmap(int16_t cab, byte value, byte fstart, byte fstop)
|
|
{
|
|
for (int16_t i = fstart; i <= fstop; i++) {
|
|
if (! DCC::setFn(cab, i, value & 1)) return false;
|
|
value >>= 1;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool DCCEXParser::execute(byte * com,Print *stream, byte opcode,byte params, int16_t p[], RingStream * ringStream) {
|
|
bool accessoryCommandReverse = false; // TODO
|
|
ZZBEGIN
|
|
ZZ(#) StringFormatter::send(stream, F("<# %d>\n"), MAX_LOCOS);
|
|
ZZ(t,cab) CHECK(cab>0)
|
|
CommandDistributor::broadcastLoco(DCC::lookupSpeedTable(cab,false));
|
|
ZZ(t,cab,tspeed,direction) CHECK(setThrottle(cab,tspeed,direction))
|
|
ZZ(t,ignore,cab,tspeed,direction) CHECK(setThrottle(cab,tspeed,direction))
|
|
// todo ZZ(f,cab,byte1) CHECK(handleFunctionGroup(cab,byte1))
|
|
// todo ZZ(f,cab,byte1,byte2) CHECK(handleFunctionGroup(cab,byte1,byte2))
|
|
|
|
ZZ(T) Turnout::printAll(stream); // will <X> if none found
|
|
ZZ(T,id) CHECK(Turnout::remove(id))
|
|
ZZ(T,id,X) auto tt=Turnout::get(id); CHECK(tt) tt->print(stream);
|
|
ZZ(T,id,T) Turnout::setClosed(id, false);
|
|
ZZ(T,id,C) Turnout::setClosed(id, true);
|
|
ZZ(T,id,value) Turnout::setClosed(id, value==0);
|
|
ZZ(T,id,SERVO,pin,low,high) CHECK(ServoTurnout::create(id, (VPIN)pin, (uint16_t)low, (uint16_t)high, 1))
|
|
ZZ(T,id,VPIN,pin) CHECK(VpinTurnout::create(id, pin))
|
|
ZZ(T,id,DCC,addr,subadd) CHECK(DCCTurnout::create(id, addr, subadd))
|
|
ZZ(T,id,DCC,nn) CHECK(DCCTurnout::create(id, (nn-1)/4+1, (nn-1)%4))
|
|
ZZ(T,id,addr,subadd) CHECK(DCCTurnout::create(id, addr, subadd))
|
|
ZZ(T,id,pin,low,high) CHECK(ServoTurnout::create(id, (VPIN)pin,low,high,1))
|
|
ZZ(S,id,pin,pullup) CHECK(Sensor::create(id,pin,pullup))
|
|
ZZ(S,id) CHECK(Sensor::remove(p[0]))
|
|
ZZ(S) for (auto *tt = Sensor::firstSensor; tt; tt = tt->nextSensor) {
|
|
StringFormatter::send(stream, F("<Q %d %d %d>\n"), tt->data.snum, tt->data.pin, tt->data.pullUp);
|
|
}
|
|
ZZ(J,M) Stash::list(stream);
|
|
ZZ(J,M,stash_id) Stash::list(stream, stash_id);
|
|
ZZ(J,M,CLEAR,ALL) Stash::clearAll();
|
|
ZZ(J,M,CLEAR,stash_id) Stash::clear(stash_id);
|
|
ZZ(J,M,stashId,locoId) Stash::set(stashId,locoId);
|
|
ZZ(J,M,CLEAR,ANY,locoId) Stash::clearAny(locoId);
|
|
ZZ(J,C) StringFormatter::send(stream, F("<jC %d>\n"), CommandDistributor::retClockTime());
|
|
ZZ(J,C,mmmm,nn) CommandDistributor::setClockTime(mmmm, nn, 1);
|
|
|
|
ZZ(J,G) TrackManager::reportGauges(stream); // <g limit...limit>
|
|
ZZ(J,I) TrackManager::reportCurrent(stream); // <g limit...limit>
|
|
ZZ(J,L,display,row) TrackManager::reportCurrentLCD(display,row); // Track power status
|
|
ZZ(J,A) StringFormatter::send(stream, F("<jA>\n")); // <JA> intercepted by EXRAIL// <JA> returns automations/routes
|
|
ZZ(J,R) StringFormatter::send(stream, F("<jR"));
|
|
SENDFLASHLIST(stream,RMFT2::rosterIdList)
|
|
StringFormatter::send(stream, F(">\n"));
|
|
ZZ(J,R,id) auto rosterName= RMFT2::getRosterName(id);
|
|
if (!rosterName) rosterName=F("");
|
|
auto functionNames= RMFT2::getRosterFunctions(id);
|
|
if (!functionNames) functionNames=RMFT2::getRosterFunctions(0);
|
|
if (!functionNames) functionNames=F("");
|
|
StringFormatter::send(stream,F(" %d \"%S\" \"%S\">\n"),
|
|
id, rosterName, functionNames);
|
|
ZZ(J,T) // <JT> returns turnout list
|
|
StringFormatter::send(stream, F("<jT"));
|
|
for ( auto t=Turnout::first(); t; t=t->next()) {
|
|
if (t->isHidden()) continue;
|
|
StringFormatter::send(stream, F(" %d"),t->getId());
|
|
}
|
|
|
|
StringFormatter::send(stream, F(">\n"));
|
|
ZZ(J,T,id) auto t=Turnout::get(id);
|
|
if (!t || t->isHidden()) StringFormatter::send(stream, F(" %d X"),id);
|
|
else {
|
|
const FSH *tdesc = RMFT2::getTurnoutDescription(id);
|
|
if (!tdesc) tdesc = F("");
|
|
StringFormatter::send(stream, F("<jT %d %c \"%S\">\n"),
|
|
id,t->isThrown()?'T':'C',
|
|
tdesc);
|
|
}
|
|
ZZ(z,vpin) // <z vpin | -vpin>
|
|
if (vpin>0) IODevice::write(vpin,HIGH);
|
|
else IODevice::write(-vpin,LOW);
|
|
ZZ(z,vpin,analog,profile,duration) IODevice::writeAnalogue(vpin,analog,profile,duration);
|
|
ZZ(z,vpin,analog,profile) IODevice::writeAnalogue(vpin,analog,profile,0);
|
|
ZZ(z,vpin,analog) IODevice::writeAnalogue(vpin,analog,0,0);
|
|
|
|
// ==========================
|
|
// Turntable - no support if no HAL
|
|
// <I> - list all
|
|
// <I id> - broadcast type and current position
|
|
// <I id DCC> - create DCC - This is TBA
|
|
// <I id steps> - operate (DCC)
|
|
// <I id steps activity> - operate (EXTT)
|
|
// <I id ADD position value> - add position
|
|
// <I id EXTT i2caddress vpin home> - create EXTT
|
|
|
|
ZZ(I) return Turntable::printAll(stream);
|
|
|
|
ZZ(I,id) // <I id> broadcast type and current position
|
|
auto tto = Turntable::get(id);
|
|
CHECK(tto)
|
|
StringFormatter::send(stream, F("<I %d %d>\n"), tto->isEXTT(), tto->getPosition());
|
|
|
|
|
|
ZZ(I,id,position) // <I id position> - rotate a DCC turntable
|
|
auto tto = Turntable::get(id);
|
|
CHECK(tto)
|
|
CHECK(!tto->isEXTT())
|
|
CHECK(tto->setPosition(id,position))
|
|
|
|
ZZ(I,id,DCC,home)
|
|
auto tto = Turntable::get(id);
|
|
CHECK(tto)
|
|
CHECK(home >=0 && home <= 3600)
|
|
CHECK(DCCTurntable::create(id))
|
|
tto = Turntable::get(id);
|
|
CHECK(tto)
|
|
tto->addPosition(0, 0, home);
|
|
StringFormatter::send(stream, F("<I>\n"));
|
|
|
|
ZZ(I,id,position,activity)
|
|
auto tto = Turntable::get(id);
|
|
CHECK(tto)
|
|
CHECK(tto->isEXTT())
|
|
CHECK(tto->setPosition(id, position,activity))
|
|
|
|
ZZ(I,id,EXTT,vpin,home) // <I id EXTT vpin home> create an EXTT turntable
|
|
auto tto = Turntable::get(id);
|
|
CHECK(!tto && home >= 0 && home <= 3600)
|
|
CHECK(EXTTTurntable::create(id, (VPIN)vpin))
|
|
tto = Turntable::get(id);
|
|
tto->addPosition(0, 0, home);
|
|
StringFormatter::send(stream, F("<I>\n"));
|
|
|
|
|
|
ZZ(I,id,ADD,position,value,angle) // <I id ADD position value angle> add a position
|
|
auto tto = Turntable::get(p[0]);
|
|
// tto must exist, no more than 48 positions, angle 0 - 3600
|
|
CHECK(tto && position <= 48 && angle >=0 && angle <= 3600)
|
|
tto->addPosition(id,value,angle);
|
|
StringFormatter::send(stream, F("<I>\n"));
|
|
|
|
ZZ(Q) Sensor::printAll(stream);
|
|
|
|
ZZ(s) // STATUS <s>
|
|
StringFormatter::send(stream, F("<iDCC-EX V-%S / %S / %S G-%S>\n"), F(VERSION), F(ARDUINO_TYPE), DCC::getMotorShieldName(), F(GITHUB_SHA));
|
|
CommandDistributor::broadcastPower(); // <s> is the only "get power status" command we have
|
|
Turnout::printAll(stream); //send all Turnout states
|
|
Sensor::printAll(stream); //send all Sensor states
|
|
|
|
|
|
#ifndef DISABLE_EEPROM
|
|
ZZ(E) // STORE EPROM <E>
|
|
EEStore::store();
|
|
StringFormatter::send(stream, F("<e %d %d %d>\n"), EEStore::eeStore->data.nTurnouts, EEStore::eeStore->data.nSensors, EEStore::eeStore->data.nOutputs);
|
|
|
|
ZZ(e) // CLEAR EPROM <e>
|
|
EEStore::clear();
|
|
StringFormatter::send(stream, F("<O>\n"));
|
|
|
|
#endif
|
|
|
|
ZZ(Z,id,active) auto o = Output::get(id);
|
|
CHECK(o)
|
|
o->activate(active);
|
|
StringFormatter::send(stream, F("<Y %d %d>\n"), id,active);
|
|
|
|
ZZ(Z,id,pin,iflag) // <Z ID PIN IFLAG>
|
|
CHECK(id > 0 && iflag >= 0 && iflag <= 7 )
|
|
CHECK(Output::create(id,pin,iflag, 1))
|
|
StringFormatter::send(stream, F("<O>\n"));
|
|
ZZ(Z,id) CHECK(Output::remove(id))
|
|
StringFormatter::send(stream, F("<O>\n"));
|
|
|
|
ZZ(Z) // <Z> list Output definitions
|
|
bool gotone = false;
|
|
for (auto *tt = Output::firstOutput; tt ; tt = tt->nextOutput){
|
|
gotone = true;
|
|
StringFormatter::send(stream, F("<Y %d %d %d %d>\n"),
|
|
tt->data.id, tt->data.pin, tt->data.flags, tt->data.active);
|
|
}
|
|
CHECK(gotone)
|
|
ZZ(D,ACK,ON) Diag::ACK = true;
|
|
ZZ(D,ACK,OFF) Diag::ACK = false;
|
|
ZZ(D,CABS) DCC::displayCabList(stream);
|
|
ZZ(D,RAM) DIAG(F("Free memory=%d"), DCCTimer::getMinimumFreeMemory());
|
|
ZZ(D,CMD,ON) Diag::CMD = true;
|
|
ZZ(D,CMD,OFF) Diag::CMD = false;
|
|
ZZ(D,RAILCOM,ON) Diag::RAILCOM = true;
|
|
ZZ(D,RAILCOM,OFF) Diag::RAILCOM = false;
|
|
ZZ(D,WIFI,ON) Diag::WIFI = true;
|
|
ZZ(D,WIFI,OFF) Diag::WIFI = false;
|
|
ZZ(D,ETHERNET,ON) Diag::ETHERNET = true;
|
|
ZZ(D,ETHERNET,OFF) Diag::ETHERNET = false;
|
|
ZZ(D,WIT,ON) Diag::WITHROTTLE = true;
|
|
ZZ(D,WIT,OFF) Diag::WITHROTTLE = false;
|
|
ZZ(D,LCN,ON) Diag::LCN = true;
|
|
ZZ(D,LCN,OFF) Diag::LCN = false;
|
|
ZZ(D,WEBSOCKET,ON) Diag::WEBSOCKET = true;
|
|
ZZ(D,WEBSOCKET,OFF) Diag::WEBSOCKET = false;
|
|
|
|
#ifndef DISABLE_EEPROM
|
|
ZZ(D,EEPROM,numentries) EEStore::dump(numentries);
|
|
#endif
|
|
|
|
|
|
ZZ(D,ANOUT,vpin,position) IODevice::writeAnalogue(vpin,position,0);
|
|
ZZ(D,ANOUT,vpin,position,profile) IODevice::writeAnalogue(vpin,position,profile);
|
|
ZZ(D,SERVO,vpin,position) IODevice::writeAnalogue(vpin,position,0);
|
|
ZZ(D,SERVO,vpin,position,profile) IODevice::writeAnalogue(vpin,position,profile);
|
|
|
|
ZZ(D,ANIN,vpin)// <D ANIN vpin> Display analogue input value
|
|
DIAG(F("VPIN=%u value=%d"), vpin, IODevice::readAnalogue(vpin));
|
|
|
|
ZZ(D,HAL,SHOW) IODevice::DumpAll();
|
|
ZZ(D,HAL,RESET) IODevice::reset();
|
|
ZZ(D,TT,vpin,steps) IODevice::writeAnalogue(vpin,steps,0);
|
|
ZZ(D,TT,vpin,steps,activity) IODevice::writeAnalogue(vpin,steps,activity);
|
|
|
|
ZZ(C,PROGBOOST) TrackManager::progTrackBoosted=true;
|
|
ZZ(C,RESET) DCCTimer::reset();
|
|
ZZ(C,SPEED28) DCC::setGlobalSpeedsteps(28); DIAG(F("28 Speedsteps"));
|
|
ZZ(C,SPEED128) DCC::setGlobalSpeedsteps(128); DIAG(F("128 Speedsteps"));
|
|
ZZ(C,RAILCOM,ON) DIAG(F("Railcom %S"),DCCWaveform::setRailcom(true,false)?F("ON"):F("OFF"));
|
|
ZZ(C,RAILCOM,OFF) DIAG(F("Railcom OFF")); DCCWaveform::setRailcom(false,false);
|
|
ZZ(C,RAILCOM,DEBUG) DIAG(F("Railcom %S"), DCCWaveform::setRailcom(true,true)?F("ON"):F("OFF"));
|
|
|
|
#ifndef DISABLE_PROG
|
|
ZZ(D,ACK,LIMIT,value) DCCACK::setAckLimit(value); LCD(1, F("Ack Limit=%dmA"), value);
|
|
ZZ(D,ACK,MIN,value,MS) DCCACK::setMinAckPulseDuration(value*1000L); LCD(1, F("Ack Min=%dmS"), value);
|
|
ZZ(D,ACK,MIN,value) DCCACK::setMinAckPulseDuration(value); LCD(1, F("Ack Min=%duS"), value);
|
|
ZZ(D,ACK,MAX,value,MS) DCCACK::setMaxAckPulseDuration(value*1000L); LCD(1, F("Ack Max=%dmS"), value);
|
|
ZZ(D,ACK,MAX,value) DCCACK::setMaxAckPulseDuration(value); LCD(1, F("Ack Max=%duS"), value);
|
|
ZZ(D,ACK,RETRY,value) DCCACK::setAckRetry(value); LCD(1, F("Ack Retry=%d"), value);
|
|
#endif
|
|
#if defined(ARDUINO_ARCH_ESP32)
|
|
// currently this only works on ESP32
|
|
ZZ(C,WIFI,marker1,ssid,marker2,password)
|
|
// <C WIFI SSID PASSWORD>
|
|
CHECK(marker1==0x7777 && marker2==0x7777)
|
|
WifiESP::setup((const char*)(com + p[2]), (const char*)(com + p[4]), WIFI_HOSTNAME, IP_PORT, WIFI_CHANNEL, WIFI_FORCE_AP);
|
|
#endif
|
|
|
|
ZZ(o,vpin) IODevice::write(abs(vpin),vpin>0);
|
|
ZZ(o,vpin,count) IODevice::writeRange(abs(vpin),vpin>0,count);
|
|
ZZ(o,vpin,r,g,b) CHECK(r>-0 && r<=0xff) CHECK(g>-0 && g<=0xff) CHECK(b>-0 && b<=0xff)
|
|
IODevice::writeAnalogueRange(abs(vpin),vpin>0,r<<8 | g,b,1);
|
|
ZZ(o,vpin,r,g,b,count) CHECK(r>-0 && r<=0xff) CHECK(g>-0 && g<=0xff) CHECK(b>-0 && b<=0xff)
|
|
IODevice::writeAnalogueRange(abs(vpin),vpin>0,r<<8 | g,b,count);
|
|
|
|
ZZ(1) TrackManager::setTrackPower(TRACK_ALL, POWERMODE::ON);
|
|
ZZ(1,MAIN) TrackManager::setTrackPower(TRACK_MODE_MAIN, POWERMODE::ON);
|
|
#ifndef DISABLE_PROG
|
|
ZZ(1,PROG) TrackManager::setJoin(false); TrackManager::setTrackPower(TRACK_MODE_PROG, POWERMODE::ON);
|
|
ZZ(1,JOIN) TrackManager::setJoin(true); TrackManager::setTrackPower(TRACK_MODE_MAIN|TRACK_MODE_PROG, POWERMODE::ON);
|
|
#endif
|
|
ZZ(1,letter) CHECK(letter>='A' && letter<='H') TrackManager::setTrackPower(POWERMODE::ON, (byte)letter-'A');
|
|
|
|
ZZ(0) TrackManager::setJoin(false); TrackManager::setTrackPower(TRACK_ALL, POWERMODE::OFF);
|
|
ZZ(0,MAIN)TrackManager::setJoin(false); TrackManager::setTrackPower(TRACK_MODE_MAIN, POWERMODE::OFF);
|
|
ZZ(0,PROG) TrackManager::setJoin(false); TrackManager::progTrackBoosted=false;
|
|
// todo move to TrackManager Prog track boost mode will not outlive prog track off
|
|
TrackManager::setTrackPower(TRACK_MODE_PROG, POWERMODE::OFF);
|
|
ZZ(0,letter) CHECK(letter>='A' && letter <='H')
|
|
TrackManager::setJoin(false);
|
|
TrackManager::setTrackPower(POWERMODE::OFF, (byte)letter-'a');
|
|
|
|
ZZ(!) DCC::estopAll(); // this broadcasts speed 1(estop) and sets all reminders to speed 1.
|
|
ZZ(c) TrackManager::reportObsoleteCurrent(stream);
|
|
|
|
ZZ(a,address,subaddress,activate) DCC::setAccessory(address, subaddress,activate ^ accessoryCommandReverse);
|
|
ZZ(a,address,subaddress,activate,onoff) CHECK(onoff>=0 && onoff<-2)
|
|
DCC::setAccessory(address, subaddress,activate ^ accessoryCommandReverse ,onoff);
|
|
ZZ(a,linearaddress,activate)
|
|
DCC::setAccessory((linearaddress - 1) / 4 + 1,(linearaddress - 1) % 4 ,activate ^ accessoryCommandReverse);
|
|
ZZ(A,address,value) DCC::setExtendedAccessory(address,value);
|
|
|
|
ZZ(w,cab,cv,value) DCC::writeCVByteMain(p[0], p[1], p[2]);
|
|
ZZ(r,cab,cv)
|
|
CHECK(DCCWaveform::isRailcom())
|
|
EXPECT_CALLBACK
|
|
DCC::readCVByteMain(cab,cv,callback_r);
|
|
ZZ(b,cab,cv,bit,value) DCC::writeCVBitMain(cab,cv,bit,value);
|
|
ZZ(m,LINEAR) DCC::linearAcceleration=true;
|
|
ZZ(m,POWER) DCC::linearAcceleration=false;
|
|
ZZ(m,cab,momentum) CHECK(DCC::setMomentum(cab,momentum,momentum))
|
|
ZZ(m,cab,momentum,braking) CHECK(DCC::setMomentum(cab,momentum,braking))
|
|
|
|
ZZ(W,cv,value,ignore1,ignore2) EXPECT_CALLBACK DCC::writeCVByte(cv,value, callback_W4);
|
|
ZZ(W,cab) EXPECT_CALLBACK DCC::setLocoId(cab,callback_Wloco);
|
|
ZZ(W,CONSIST,cab,REVERSE) EXPECT_CALLBACK DCC::setConsistId(cab,true,callback_Wconsist);
|
|
ZZ(W,CONSIST,cab) EXPECT_CALLBACK DCC::setConsistId(cab,false,callback_Wconsist);
|
|
ZZ(W,cv,value) EXPECT_CALLBACK DCC::writeCVByte(cv,value, callback_W);
|
|
ZZ(W,cv,value,bit) EXPECT_CALLBACK DCC::writeCVBit(cv,value,bit,callback_W);
|
|
ZZ(V,cv,value) EXPECT_CALLBACK DCC::verifyCVByte(cv,value, callback_Vbyte);
|
|
ZZ(V,cv,bit,value) EXPECT_CALLBACK DCC::verifyCVBit(cv,bit,value,callback_Vbit);
|
|
|
|
ZZ(B,cv,bit,value) EXPECT_CALLBACK DCC::writeCVBit(cv,bit,value,callback_B);
|
|
ZZ(R,cv,ignore1,ignore2) EXPECT_CALLBACK DCC::readCV(cv,callback_R);
|
|
ZZ(R,cv) EXPECT_CALLBACK DCC::verifyCVByte(cv, 0, callback_Vbyte);
|
|
ZZ(R) EXPECT_CALLBACK DCC::getLocoId(callback_Rloco);
|
|
|
|
#ifndef DISABLE_VDPY
|
|
ZZ(@) CommandDistributor::setVirtualLCDSerial(stream);
|
|
StringFormatter::send(stream,
|
|
F("<@ 0 0 \"DCC-EX v" VERSION "\">\n"
|
|
"<@ 0 1 \"Lic GPLv3\">\n"));
|
|
#endif
|
|
|
|
ZZ(-) DCC::forgetAllLocos();
|
|
ZZ(-,cab) DCC::forgetLoco(cab);
|
|
ZZ(F,cab,DCCFREQ,value) CHECK(value>=0 && value<=3) DCC::setDCFreq(cab,value);
|
|
ZZ(F,cab,function,value) CHECK(value==0 || value==1) DCC::setFn(cab,function,value);
|
|
|
|
|
|
ZZ(M,ignore,d0,d1,d2,d3,d4,d5) byte packet[]={(byte)d0,(byte)d1,(byte)d2,(byte)d3,(byte)d4,(byte)d5}; DCCWaveform::mainTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(M,ignore,d0,d1,d2,d3,d4) byte packet[]={(byte)d0,(byte)d1,(byte)d2,(byte)d3,(byte)d4}; DCCWaveform::mainTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(M,ignore,d0,d1,d2,d3) byte packet[]={(byte)d0,(byte)d1,(byte)d2,(byte)d3}; DCCWaveform::mainTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(M,ignore,d0,d1,d2) byte packet[]={(byte)d0,(byte)d1,(byte)d2}; DCCWaveform::mainTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(M,ignore,d0,d1) byte packet[]={(byte)d0,(byte)d1}; DCCWaveform::mainTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(P,ignore,d0,d1,d2,d3,d4,d5) byte packet[]={(byte)d0,(byte)d1,(byte)d2,(byte)d3,(byte)d4,(byte)d5}; DCCWaveform::progTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(P,ignore,d0,d1,d2,d3,d4) byte packet[]={(byte)d0,(byte)d1,(byte)d2,(byte)d3,(byte)d4}; DCCWaveform::progTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(P,ignore,d0,d1,d2,d3) byte packet[]={(byte)d0,(byte)d1,(byte)d2,(byte)d3}; DCCWaveform::progTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(P,ignore,d0,d1,d2) byte packet[]={(byte)d0,(byte)d1,(byte)d2}; DCCWaveform::progTrack.schedulePacket(packet,sizeof(packet),3);
|
|
ZZ(P,ignore,d0,d1) byte packet[]={(byte)d0,(byte)d1}; DCCWaveform::progTrack.schedulePacket(packet,sizeof(packet),3);
|
|
|
|
ZZ(J,O) StringFormatter::send(stream, F("<jO"));
|
|
for (auto tto=Turntable::first(); tto; tto=tto->next()) {
|
|
if (!tto->isHidden()) StringFormatter::send(stream, F(" %d"),tto->getId());
|
|
}
|
|
StringFormatter::send(stream, F(">\n"));
|
|
ZZ(J,O,id) auto tto=Turntable::get(id);
|
|
if (!tto || tto->isHidden()) {StringFormatter::send(stream, F("<jO %d X>\n"), id); return true;}
|
|
const FSH *todesc = nullptr;
|
|
#ifdef EXRAIL_ACTIVE
|
|
todesc = RMFT2::getTurntableDescription(id);
|
|
#endif
|
|
if (todesc == nullptr) todesc = F("");
|
|
StringFormatter::send(stream, F("<jO %d %d %d %d \"%S\">\n"), id, tto->isEXTT(), tto->getPosition(), tto->getPositionCount(), todesc);
|
|
|
|
ZZ(J,P,id) auto tto=Turntable::get(id);
|
|
if (!tto || tto->isHidden()) {StringFormatter::send(stream, F("<jP %d X>\n"), id); return true;}
|
|
auto posCount = tto->getPositionCount();
|
|
if (posCount==0) {StringFormatter::send(stream, F("<jP X>\n"));return true;}
|
|
|
|
for (auto p = 0; p < posCount; p++) {
|
|
const FSH *tpdesc = nullptr;
|
|
#ifdef EXRAIL_ACTIVE
|
|
tpdesc = RMFT2::getTurntablePositionDescription(id, p);
|
|
#endif
|
|
if (tpdesc == NULL) tpdesc = F("");
|
|
StringFormatter::send(stream, F("<jP %d %d %d \"%S\">\n"), id, p, tto->getPositionAngle(p), tpdesc);
|
|
}
|
|
|
|
|
|
ZZEND
|
|
|
|
}
|
|
|
|
// CALLBACKS must be static
|
|
bool DCCEXParser::stashCallback(Print *stream, int16_t p[MAX_COMMAND_PARAMS], RingStream * ringStream)
|
|
{
|
|
if (stashBusy )
|
|
return false;
|
|
stashBusy = true;
|
|
stashStream = stream;
|
|
stashRingStream=ringStream;
|
|
if (ringStream) stashTarget= ringStream->peekTargetMark();
|
|
memcpy(stashP, p, MAX_COMMAND_PARAMS * sizeof(p[0]));
|
|
return true;
|
|
}
|
|
|
|
Print * DCCEXParser::getAsyncReplyStream() {
|
|
if (stashRingStream) {
|
|
stashRingStream->mark(stashTarget);
|
|
return stashRingStream;
|
|
}
|
|
return stashStream;
|
|
}
|
|
|
|
void DCCEXParser::commitAsyncReplyStream() {
|
|
if (stashRingStream) stashRingStream->commit();
|
|
stashBusy = false;
|
|
}
|
|
|
|
void DCCEXParser::callback_W(int16_t result)
|
|
{
|
|
StringFormatter::send(getAsyncReplyStream(),
|
|
F("<r %d %d>\n"), stashP[0], result == 1 ? stashP[1] : -1);
|
|
commitAsyncReplyStream();
|
|
}
|
|
|
|
void DCCEXParser::callback_W4(int16_t result)
|
|
{
|
|
StringFormatter::send(getAsyncReplyStream(),
|
|
F("<r%d|%d|%d %d>\n"), stashP[2], stashP[3], stashP[0], result == 1 ? stashP[1] : -1);
|
|
commitAsyncReplyStream();
|
|
}
|
|
|
|
void DCCEXParser::callback_B(int16_t result)
|
|
{
|
|
StringFormatter::send(getAsyncReplyStream(),
|
|
F("<r%d|%d|%d %d %d>\n"), stashP[3], stashP[4], stashP[0], stashP[1], result == 1 ? stashP[2] : -1);
|
|
commitAsyncReplyStream();
|
|
}
|
|
void DCCEXParser::callback_Vbit(int16_t result)
|
|
{
|
|
StringFormatter::send(getAsyncReplyStream(), F("<v %d %d %d>\n"), stashP[0], stashP[1], result);
|
|
commitAsyncReplyStream();
|
|
}
|
|
void DCCEXParser::callback_Vbyte(int16_t result)
|
|
{
|
|
StringFormatter::send(getAsyncReplyStream(), F("<v %d %d>\n"), stashP[0], result);
|
|
commitAsyncReplyStream();
|
|
}
|
|
|
|
void DCCEXParser::callback_R(int16_t result)
|
|
{
|
|
StringFormatter::send(getAsyncReplyStream(), F("<r%d|%d|%d %d>\n"), stashP[1], stashP[2], stashP[0], result);
|
|
commitAsyncReplyStream();
|
|
}
|
|
|
|
void DCCEXParser::callback_r(int16_t result)
|
|
{
|
|
StringFormatter::send(getAsyncReplyStream(), F("<r %d %d %d >\n"), stashP[0], stashP[1], result);
|
|
commitAsyncReplyStream();
|
|
}
|
|
|
|
void DCCEXParser::callback_Rloco(int16_t result) {
|
|
const FSH * detail;
|
|
if (result<=0) {
|
|
detail=F("<r %d>\n");
|
|
} else {
|
|
bool longAddr=result & LONG_ADDR_MARKER; //long addr
|
|
if (longAddr)
|
|
result = result^LONG_ADDR_MARKER;
|
|
if (longAddr && result <= HIGHEST_SHORT_ADDR)
|
|
detail=F("<r LONG %d UNSUPPORTED>\n");
|
|
else
|
|
detail=F("<r %d>\n");
|
|
}
|
|
StringFormatter::send(getAsyncReplyStream(), detail, result);
|
|
commitAsyncReplyStream();
|
|
}
|
|
|
|
void DCCEXParser::callback_Wloco(int16_t result)
|
|
{
|
|
if (result==1) result=stashP[0]; // pick up original requested id from command
|
|
StringFormatter::send(getAsyncReplyStream(), F("<w %d>\n"), result);
|
|
commitAsyncReplyStream();
|
|
}
|
|
|
|
void DCCEXParser::callback_Wconsist(int16_t result)
|
|
{
|
|
if (result==1) result=stashP[1]; // pick up original requested id from command
|
|
StringFormatter::send(getAsyncReplyStream(), F("<w CONSIST %d%S>\n"),
|
|
result, stashP[2]=="REVERSE"_hk ? F(" REVERSE") : F(""));
|
|
commitAsyncReplyStream();
|
|
}
|