forked from chrisstaite/TeensyDmx
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathTeensyDmx.h
239 lines (207 loc) · 8.18 KB
/
TeensyDmx.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
/* TeensyDmx - DMX Sender/Receiver with RDM for Teensy 3.2
Copyright (c) 2017-2018 Peter Newman, Dan Large, http://daniellarge.co.uk
Copyright (c) 2017 Chris Staite
Copyright (c) 2014 Jim Paris
Copyright (c) 2014 Ward
Copyright (c) 2008-2009 Peter Knight, Tinker.it! All rights reserved.
Copyright (c) 2011-2013 by Matthias Hertel, http://www.mathertel.de
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.
*/
#ifndef _TEENSYDMX_H
#define _TEENSYDMX_H
#include "Arduino.h"
enum { DMX_BUFFER_SIZE = 512 };
enum { RDM_UID_LENGTH = 6 };
enum { RDM_MAX_STRING_LENGTH = 32 };
enum { RDM_MAX_PARAMETER_DATA_LENGTH = 231 };
enum { RDM_ROOT_DEVICE = 0 };
struct RdmData
{
byte startCode; // Start Code 0xCC for RDM
byte subStartCode; // Sub Start Code 0x01 for RDM
byte length; // packet length
byte destId[RDM_UID_LENGTH];
byte sourceId[RDM_UID_LENGTH];
byte transNo; // transaction number, not checked
byte responseType; // ResponseType or PortID
byte messageCount; // number of queued messages
uint16_t subDev; // sub device number (root = 0)
byte cmdClass; // command class
uint16_t parameter; // parameter ID
byte dataLength; // parameter data length in bytes
byte data[RDM_MAX_PARAMETER_DATA_LENGTH]; // data byte field
} __attribute__((__packed__)); // struct RdmData
static_assert((sizeof(RdmData) == 255),
"Invalid size for RdmData struct, is it packed?");
struct RdmInit
{
const byte *uid;
const uint32_t softwareVersionId;
const char *softwareLabel;
const char *manufacturerLabel;
const uint16_t deviceModelId;
const char *deviceModel;
const uint16_t productCategory;
uint16_t footprint;
uint16_t startAddress;
const uint16_t additionalCommandsLength;
const uint16_t *additionalCommands;
};
class TeensyDmx
{
public:
enum Mode { DMX_OFF, DMX_IN, DMX_OUT };
TeensyDmx(HardwareSerial& uart, struct RdmInit* rdm, uint8_t redePin);
TeensyDmx(HardwareSerial& uart, struct RdmInit* rdm);
TeensyDmx(HardwareSerial& uart, uint8_t redePin) :
TeensyDmx(uart, nullptr, redePin)
{ }
TeensyDmx(HardwareSerial& uart) :
TeensyDmx(uart, nullptr)
{ }
void setMode(TeensyDmx::Mode mode);
void loop();
// Returns true if a new frame has been received since the this was last called
bool newFrame();
// Get the buffer with the current channel data in
const volatile uint8_t* getBuffer() const;
// Use for receive with addresses from 0-511
uint8_t getChannel(const uint16_t address);
// Use for receive with addresses from 1-512
uint8_t getDmxChannel(const uint16_t address)
{
return getChannel(address - 1);
}
// Returns true if RDM has changed since this was last called
bool rdmChanged();
// Returns true if the device should be in identify mode
bool isIdentify() const;
// Returns the user-set label of the device
// This will always be null terminated, therefore can be up to 33 chars
const char* getLabel() const;
// Return the comms status error counters
const volatile uint16_t getShortMessage() const;
const volatile uint16_t getChecksumFail() const;
const volatile uint16_t getLengthMismatch() const;
// Use for transmit with addresses from 0-511
// Will keep all other values as they were previously
void setChannel(const uint16_t address, const uint8_t value);
// Use for transmit with addresses from 1-512
// Will keep all other values as they were previously
void setDmxChannel(const uint16_t address, const uint8_t value)
{
setChannel(address - 1, value);
}
// Use for transmit with channels from 0-511
// Will set all other channels to 0
void setChannels(const uint16_t startAddress, const uint8_t* values, const uint16_t length);
// Use for transmit with channels from 1-512
// Will set all other channels to 0
void setDmxChannels(const uint16_t startAddress, const uint8_t* values, const uint16_t length)
{
setChannels(startAddress - 1, values, length);
}
private:
TeensyDmx(const TeensyDmx&);
TeensyDmx& operator=(const TeensyDmx&);
enum State { IDLE, BREAK, DMX_TX, DMX_RECV, DMX_COMPLETE, RDM_RECV, RDM_RECV_CHECKSUM_HI, RDM_RECV_CHECKSUM_LO, RDM_RECV_POST_CHECKSUM };
void startTransmit();
void stopTransmit();
void startReceive();
void stopReceive();
void setDirection(bool transmit);
void completeFrame(); // Called at error ISR during recv
void processRDM();
void respondMessage(uint16_t nackReason);
void handleByte(uint8_t c);
void nextTx();
// RDM handler functions
void rdmDiscUniqueBranch();
uint16_t rdmDiscMute();
uint16_t rdmDiscUnMute();
uint16_t rdmGetCommsStatus();
uint16_t rdmSetCommsStatus();
uint16_t rdmGetDeviceInfo();
uint16_t rdmGetDeviceLabel();
uint16_t rdmSetDeviceLabel();
uint16_t rdmGetDeviceModelDescription();
uint16_t rdmGetDMXStartAddress();
uint16_t rdmSetDMXStartAddress();
uint16_t rdmGetIdentifyDevice();
uint16_t rdmSetIdentifyDevice();
uint16_t rdmGetManufacturerLabel();
uint16_t rdmGetSoftwareVersionLabel();
uint16_t rdmGetSupportedParameters();
uint16_t rdmCalculateChecksum(uint8_t* data, uint8_t length);
bool isForMe(const byte* id);
bool isForVendor(const byte* id);
bool isForAll(const byte* id);
void maybeIncrementShortMessage();
void maybeIncrementChecksumFail();
void maybeIncrementLengthMismatch();
HardwareSerial& m_uart;
volatile uint8_t m_dmxBuffer1[DMX_BUFFER_SIZE];
volatile uint8_t m_dmxBuffer2[DMX_BUFFER_SIZE];
volatile uint8_t *m_activeBuffer;
volatile uint8_t *m_inactiveBuffer;
volatile uint16_t m_dmxBufferIndex;
volatile unsigned int m_frameCount;
volatile uint16_t m_shortMessage;
volatile uint16_t m_checksumFail;
volatile uint16_t m_lengthMismatch;
volatile bool m_newFrame;
volatile bool m_rdmChange;
Mode m_mode;
State m_state;
volatile uint8_t* m_redePin;
bool m_rdmMute;
bool m_identifyMode;
RdmInit *m_rdm;
volatile bool m_rdmNeedsProcessing;
RdmData m_rdmBuffer;
uint16_t m_rdmChecksum;
// Allow an extra byte for a null if we have a 32 character string
char m_deviceLabel[RDM_MAX_STRING_LENGTH + 1];
static_assert((sizeof(m_deviceLabel) == 33), "Invalid size for m_deviceLabel");
friend void UART0RxStatus(void);
friend void UART0TxStatus(void);
friend void UART1RxStatus(void);
friend void UART1TxStatus(void);
friend void UART2RxStatus(void);
friend void UART2TxStatus(void);
#ifndef IRQ_UART0_ERROR
friend void UART0RxError(void);
friend void UART1RxError(void);
friend void UART2RxError(void);
#endif
#ifdef HAS_KINETISK_UART3
friend void UART3RxStatus(void);
friend void UART3TxStatus(void);
friend void UART3RxError(void);
#endif
#ifdef HAS_KINETISK_UART4
friend void UART4RxStatus(void);
friend void UART4TxStatus(void);
friend void UART4RxError(void);
#endif
#ifdef HAS_KINETISK_UART5
friend void UART5RxStatus(void);
friend void UART5TxStatus(void);
friend void UART5RxError(void);
#endif
};
#endif // _TEENSYDMX_H