2020-07-03 18:35:02 +02:00
|
|
|
/*
|
|
|
|
* © 2020, Chris Harlow. All rights reserved.
|
|
|
|
*
|
|
|
|
* This file is part of Asbelos DCC 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 <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2020-05-27 10:40:12 +02:00
|
|
|
#ifndef DCC_h
|
|
|
|
#define DCC_h
|
2020-05-24 17:07:16 +02:00
|
|
|
#include <Arduino.h>
|
2020-09-22 23:24:30 +02:00
|
|
|
#include "config.h"
|
2020-08-15 12:32:32 +02:00
|
|
|
#include "MotorDriver.h"
|
2020-08-19 14:12:39 +02:00
|
|
|
#include "MotorDrivers.h"
|
2020-05-27 10:40:12 +02:00
|
|
|
|
2020-06-07 14:48:42 +02:00
|
|
|
typedef void (*ACK_CALLBACK)(int result);
|
|
|
|
|
|
|
|
enum ackOp { // Program opcodes for the ack Manager
|
2020-06-07 17:29:53 +02:00
|
|
|
BASELINE, // ensure enough resets sent before starting and obtain baseline current
|
|
|
|
W0,W1, // issue write bit (0..1) packet
|
|
|
|
WB, // issue write byte packet
|
2020-06-07 14:48:42 +02:00
|
|
|
VB, // Issue validate Byte packet
|
|
|
|
V0, // Issue validate bit=0 packet
|
|
|
|
V1, // issue validate bit=1 packlet
|
|
|
|
WACK, // wait for ack (or absence of ack)
|
|
|
|
ITC1, // If True Callback(1) (if prevous WACK got an ACK)
|
|
|
|
ITC0, // If True callback(0);
|
2020-06-07 17:29:53 +02:00
|
|
|
ITCB, // If True callback(byte)
|
2020-06-08 14:04:47 +02:00
|
|
|
NAKFAIL, // if false callback(-1)
|
2020-06-07 14:48:42 +02:00
|
|
|
FAIL, // callback(-1)
|
2020-06-07 17:29:53 +02:00
|
|
|
STARTMERGE, // Clear bit and byte settings ready for merge pass
|
2020-06-08 14:04:47 +02:00
|
|
|
MERGE, // Merge previous wack response with byte value and decrement bit number (use for readimng CV bytes)
|
|
|
|
SETBIT, // sets bit number to next prog byte
|
|
|
|
SETCV, // sets cv number to next prog byte
|
|
|
|
STASHLOCOID, // keeps current byte value for later
|
|
|
|
COMBINELOCOID, // combines current value with stashed value and returns it
|
|
|
|
ITSKIP, // skip to SKIPTARGET if ack true
|
|
|
|
SKIPTARGET=0xFF // jump to target
|
2020-06-07 14:48:42 +02:00
|
|
|
};
|
2020-05-24 17:07:16 +02:00
|
|
|
|
2020-08-19 14:12:39 +02:00
|
|
|
// Allocations with memory implications..!
|
|
|
|
// Base system takes approx 900 bytes + 8 per loco. Turnouts, Sensors etc are dynamically created
|
|
|
|
#ifdef ARDUINO_AVR_UNO
|
2020-09-22 23:24:30 +02:00
|
|
|
const byte MAX_LOCOS=MAX_MAIN_REGISTERS;
|
2020-08-19 14:12:39 +02:00
|
|
|
#else
|
|
|
|
const byte MAX_LOCOS=50;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2020-05-24 17:07:16 +02:00
|
|
|
class DCC {
|
|
|
|
public:
|
|
|
|
|
2020-09-24 10:51:09 +02:00
|
|
|
static void begin(const __FlashStringHelper* motorShieldName, MotorDriver * mainDriver, MotorDriver * progDriver, byte timerNumber=1);
|
2020-05-24 17:07:16 +02:00
|
|
|
static void loop();
|
|
|
|
|
|
|
|
// Public DCC API functions
|
|
|
|
static void setThrottle( uint16_t cab, uint8_t tSpeed, bool tDirection);
|
2020-06-27 16:36:32 +02:00
|
|
|
static uint8_t getThrottleSpeed(int cab);
|
|
|
|
static bool getThrottleDirection(int cab);
|
2020-05-24 17:07:16 +02:00
|
|
|
static void writeCVByteMain(int cab, int cv, byte bValue);
|
|
|
|
static void writeCVBitMain(int cab, int cv, byte bNum, bool bValue);
|
|
|
|
static void setFunction( int cab, byte fByte, byte eByte);
|
2020-06-22 11:54:57 +02:00
|
|
|
static void setFn( int cab, byte functionNumber, bool on);
|
2020-08-01 14:06:39 +02:00
|
|
|
static int changeFn( int cab, byte functionNumber, bool pressed);
|
|
|
|
static void updateGroupflags(byte & flags, int functionNumber);
|
2020-05-24 17:07:16 +02:00
|
|
|
static void setAccessory(int aAdd, byte aNum, bool activate) ;
|
|
|
|
static bool writeTextPacket( byte *b, int nBytes);
|
2020-07-12 01:11:30 +02:00
|
|
|
static void setProgTrackSyncMain(bool on); // when true, prog track becomes driveable
|
2020-06-07 14:48:42 +02:00
|
|
|
|
|
|
|
// ACKable progtrack calls bitresults callback 0,0 or -1, cv returns value or -1
|
2020-07-19 19:39:08 +02:00
|
|
|
static void readCV(int cv, ACK_CALLBACK callback, bool blocking=false);
|
|
|
|
static void readCVBit(int cv, byte bitNum, ACK_CALLBACK callback, bool blocking=false); // -1 for error
|
|
|
|
static void writeCVByte(int cv, byte byteValue, ACK_CALLBACK callback, bool blocking=false) ;
|
|
|
|
static void writeCVBit(int cv, byte bitNum, bool bitValue, ACK_CALLBACK callback, bool blocking=false);
|
2020-09-18 01:04:42 +02:00
|
|
|
static void verifyCVByte(int cv, byte byteValue, ACK_CALLBACK callback, bool blocking=false) ;
|
|
|
|
static void verifyCVBit(int cv, byte bitNum, bool bitValue, ACK_CALLBACK callback, bool blocking=false);
|
2020-06-07 14:48:42 +02:00
|
|
|
|
2020-07-19 19:39:08 +02:00
|
|
|
static void getLocoId(ACK_CALLBACK callback, bool blocking=false);
|
2020-05-24 17:07:16 +02:00
|
|
|
|
2020-06-18 20:36:37 +02:00
|
|
|
// Enhanced API functions
|
|
|
|
static void forgetLoco(int cab); // removes any speed reminders for this loco
|
|
|
|
static void forgetAllLocos(); // removes all speed reminders
|
2020-09-10 14:09:32 +02:00
|
|
|
static void displayCabList(Print * stream);
|
2020-09-24 10:51:09 +02:00
|
|
|
|
|
|
|
static __FlashStringHelper* getMotorShieldName();
|
|
|
|
|
2020-05-24 17:07:16 +02:00
|
|
|
private:
|
|
|
|
struct LOCO {
|
|
|
|
int loco;
|
2020-06-03 11:36:01 +02:00
|
|
|
byte speedCode;
|
2020-06-22 11:54:57 +02:00
|
|
|
byte groupFlags;
|
2020-08-01 14:06:39 +02:00
|
|
|
unsigned long functions;
|
2020-05-24 17:07:16 +02:00
|
|
|
};
|
2020-06-22 11:54:57 +02:00
|
|
|
static byte loopStatus;
|
2020-06-03 11:36:01 +02:00
|
|
|
static void setThrottle2( uint16_t cab, uint8_t speedCode);
|
|
|
|
static void updateLocoReminder(int loco, byte speedCode);
|
2020-06-22 11:54:57 +02:00
|
|
|
static void setFunctionInternal( int cab, byte fByte, byte eByte);
|
|
|
|
static bool issueReminder(int reg);
|
2020-05-24 17:07:16 +02:00
|
|
|
static int nextLoco;
|
2020-09-24 10:51:09 +02:00
|
|
|
static __FlashStringHelper* shieldName;
|
|
|
|
|
2020-05-24 17:07:16 +02:00
|
|
|
static LOCO speedTable[MAX_LOCOS];
|
|
|
|
static byte cv1(byte opcode, int cv);
|
|
|
|
static byte cv2(int cv);
|
2020-06-22 11:54:57 +02:00
|
|
|
static int lookupSpeedTable(int locoId);
|
|
|
|
static void issueReminders();
|
2020-07-01 11:27:53 +02:00
|
|
|
static void callback(int value);
|
|
|
|
|
2020-06-07 14:48:42 +02:00
|
|
|
// ACK MANAGER
|
|
|
|
static ackOp const * ackManagerProg;
|
|
|
|
static byte ackManagerByte;
|
|
|
|
static byte ackManagerBitNum;
|
|
|
|
static int ackManagerCv;
|
2020-06-08 14:04:47 +02:00
|
|
|
static byte ackManagerStash;
|
2020-06-07 14:48:42 +02:00
|
|
|
static bool ackReceived;
|
|
|
|
static ACK_CALLBACK ackManagerCallback;
|
2020-07-19 19:39:08 +02:00
|
|
|
static void ackManagerSetup(int cv, byte bitNumOrbyteValue, ackOp const program[], ACK_CALLBACK callback, bool blocking);
|
|
|
|
static void ackManagerLoop(bool blocking);
|
2020-09-08 09:47:40 +02:00
|
|
|
static bool checkResets(bool blocking, uint8_t numResets);
|
2020-07-11 10:35:57 +02:00
|
|
|
static const int PROG_REPEATS=8; // repeats of programming commands (some decoders need at least 8 to be reliable)
|
2020-06-07 14:48:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
2020-06-01 14:56:02 +02:00
|
|
|
// NMRA codes #
|
|
|
|
static const byte SET_SPEED=0x3f;
|
|
|
|
static const byte WRITE_BYTE_MAIN = 0xEC;
|
|
|
|
static const byte WRITE_BIT_MAIN = 0xE8;
|
|
|
|
static const byte WRITE_BYTE = 0x7C;
|
|
|
|
static const byte VERIFY_BYTE= 0x74;
|
|
|
|
static const byte BIT_MANIPULATE=0x78;
|
|
|
|
static const byte WRITE_BIT=0xF0;
|
|
|
|
static const byte VERIFY_BIT=0xE0;
|
|
|
|
static const byte BIT_ON=0x08;
|
|
|
|
static const byte BIT_OFF=0x00;
|
2020-05-24 17:07:16 +02:00
|
|
|
};
|
2020-06-07 14:48:42 +02:00
|
|
|
|
2020-09-23 21:17:01 +02:00
|
|
|
#ifdef ARDUINO_AVR_MEGA // is using Mega 1280, define as Mega 2560 (pinouts and functionality are identical)
|
|
|
|
#define ARDUINO_AVR_MEGA2560
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(ARDUINO_AVR_UNO)
|
|
|
|
#define ARDUINO_TYPE "UNO"
|
|
|
|
#elif defined(ARDUINO_AVR_NANO)
|
|
|
|
#define ARDUINO_TYPE "NANO"
|
|
|
|
#elif defined(ARDUINO_AVR_MEGA2560)
|
|
|
|
#define ARDUINO_TYPE "MEGA"
|
|
|
|
#else
|
|
|
|
#error CANNOT COMPILE - DCC++ EX ONLY WORKS WITH AN ARDUINO UNO, NANO 328, OR ARDUINO MEGA 1280/2560
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2020-09-24 03:54:01 +02:00
|
|
|
#ifdef ENABLE_LCD
|
|
|
|
#include <Wire.h>
|
|
|
|
#if defined(LIB_TYPE_PCF8574)
|
|
|
|
#include <LiquidCrystal_PCF8574.h>
|
|
|
|
extern LiquidCrystal_PCF8574 lcdDisplay;
|
|
|
|
#elif defined(LIB_TYPE_I2C)
|
|
|
|
#include <LiquidCrystal_I2C.h>
|
|
|
|
extern LiquidCrystal_I2C lcdDisplay;
|
|
|
|
#endif
|
|
|
|
extern bool lcdEnabled;
|
|
|
|
#endif
|
2020-09-23 21:17:01 +02:00
|
|
|
|
2020-05-27 10:40:12 +02:00
|
|
|
#endif
|