-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSmallNMEA2000.h
653 lines (575 loc) · 22.3 KB
/
SmallNMEA2000.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
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
#ifndef SmallNMEA2000_H
#define SmallNMEA2000_H
#if defined(__GNUC__) && defined (__BYTE_ORDER__)
#if __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__
#error "This code assumes little endian processors, CAN is little endian. byteswap patches will be required."
#endif
#endif
#include <Arduino.h>
#include <mcp_can.h>
#define CToKelvin(x) (x+273.15)
typedef struct MsgHeader {
unsigned char Priority;
unsigned long PGN;
unsigned char Source;
unsigned char Destination;
} MsgHeader;
/**
Example of the product Information buffer and configuration information buffer.
This byte[] is sent as is in fastPacket messages.
byte productInformationBuffer[] PROGMEM = {
N2K_VERSION_2BYTEINT,
PRODUCT_CODE_2BYTEINT,
MODEL_ID_BYTEFF, 0xff,
SWCODE_BYTEFF, 0xff,
MODEL_VERSION_BYTEFF, 0xff,
console->CODE_BYTEFF, 0xff,
CERTIFICATION_LEVEL_BYTE,
LOAD_EQUIV_BYTE
};
byte configurationInformation[] PROGMEM = {
INSTALLER_DESC1_LEN_P2_BYTE,
0x01,
INSTALLER_DESC1_BYTES,
0x0ff,
INSTALLER_DESC2_LEN_P2_BYTE
0x01,
INSTALLER_DESC2_BYTES,
0x0ff,
MAN_INFO_LEN_P2_BYTE
0x01,
MAN_INFO_BYTES,
0x0ff
};
*/
/**
* Appled these PGNs to your TX and RX lists when defining in PROGMEM,
* They are RX and TX by the SNMEA2000 class but we need to get them into
* a contiguous memory space in program memory.
*/
#define SNMEA200_DEFAULT_TX_PGN 126464L,60928L,126996L,126998L,59392L,0L
#define SNMEA200_DEFAULT_RX_PGN 59392L,59904L,60928L,0L
// from NMEA2000 library, makes it much easier creating the name.
// CAN and this code is Little endian so the name and order of bytes is the same.
// This union can be send directly as a message.
typedef union {
uint64_t name;
struct {
uint32_t unicNumberAndManCode; // ManufacturerCode 11 bits , UniqueNumber 21 bits
unsigned char deviceInstance; // indentifies multiple idendical devices on the same bus, eg 2 engine controllers.
unsigned char deviceFunction; // http://www.nmea.org/Assets/20120726%20nmea%202000%20class%20&%20function%20codes%20v%202.00.pdf
unsigned char deviceClass; // http://www.nmea.org/Assets/20120726%20nmea%202000%20class%20&%20function%20codes%20v%202.00.pdf
// I found document: http://www.novatel.com/assets/Documents/Bulletins/apn050.pdf it says about next fields:
// The System Instance Field can be utilized to facilitate multiple NMEA 2000 networks on these larger marine platforms.
// NMEA 2000 devices behind a bridge, router, gateway, or as part of some network segment could all indicate this by use
// and application of the System Instance Field.
// DeviceInstance and SystemInstance fields can be now changed by function SetDeviceInformationInstances or
// by NMEA 2000 group function. Group function handling is build in the library.
unsigned char industryGroupAndSystemInstance; // 4 bits each
};
} tUnionDeviceInformation;
// From https://github.com/ttlappalainen/NMEA2000/blob/ad30dced133cf7063b97aaa9ea05e434912e9100/src/NMEA2000.h#L155
class SNMEA2000DeviceInfo {
public:
/**
* @brief Construct a new Device Information object
*
* @param uniqueNumber unique mumber
* @param deviceFunction http://www.nmea.org/Assets/20120726%20nmea%202000%20class%20&%20function%20codes%20v%202.00.pdf
* @param deviceClass http://www.nmea.org/Assets/20120726%20nmea%202000%20class%20&%20function%20codes%20v%202.00.pdf
* @param manufacturersCode http://www.nmea.org/Assets/20121020%20nmea%202000%20registration%20list.pdf
* @param industryGroup 4 is marine
* @param deviceInstance
* @param systemInstance
*/
SNMEA2000DeviceInfo(
uint32_t uniqueNumber,
unsigned char deviceFunction,
unsigned char deviceClass,
uint16_t manufacturersCode = 2048,
unsigned char industryGroup = 4,
unsigned char deviceInstance = 0,
unsigned char systemInstance = 0
) {
deviceInformation.unicNumberAndManCode = (uniqueNumber&0x1fffff) | (((unsigned long)(manufacturersCode&0x7ff))<<21);
deviceInformation.deviceInstance = deviceInstance;
deviceInformation.deviceFunction = deviceFunction;
deviceInformation.deviceClass = (deviceClass&0x7f)<<1;
deviceInformation.industryGroupAndSystemInstance = 0x80 | ((industryGroup&0x07)<<4) | (systemInstance&0x0f);
};
void setSerialNumber(uint32_t uniqueNumber) {
deviceInformation.unicNumberAndManCode=(deviceInformation.unicNumberAndManCode&0xffe00000) | (uniqueNumber&0x1fffff);
};
void setDeviceInstanceNumber(uint8_t deviceInstance) {
deviceInformation.deviceInstance = deviceInstance;
};
uint64_t getName() {
return deviceInformation.name;
};
byte * getDeviceNameBuffer() {
return (byte * ) &(deviceInformation);
}
protected:
tUnionDeviceInformation deviceInformation;
};
typedef struct SNMEA2000ProductInfo {
uint16_t nk2version;
uint16_t productCode;
const char * modelID; // 32 bytes max
const char * softwareVersion; // 32 bytes max padded
const char * modelVersion; // 32 bytes max padded
const char * serialNumber; // 32 bytes max padded
byte certificationLevel;
byte loadEquivalency;
} SNMEA2000ProductInfo;
typedef struct SNMEA2000ConfigInfo {
const char * manufacturerInfo; // max 71 var string
const char * installDesc1; // max 71 var string
const char * installDesc2; // max 71 var string
} SNMEA2000ConfigInfo;
class MessageHeader {
public:
MessageHeader(unsigned long ID) {
// CAN ID is 29 bits long
// 111 1 1 11111111 11111111 11111111
// -------- Source Addresss 8 bits @0
// -------- PDU specific 8 bits @ 8
// -------- PDU Format 8 bits @ 16
// - Data Page 1 but @24
// - reserved 1 bit @25
// --- priority (3 bits starting @26)
// PDU Format < 240 is transmitted with a destination address, which is in 8 bits 8
// PDU Format 240 > 255 is a broadcast with the Group extension in 8 bits @8.
id = ID;
unsigned char canIdPF = (unsigned char) (id >> 16);
unsigned char canIdPS = (unsigned char) (id >> 8);
unsigned char canIdDP = (unsigned char) (id >> 24) & 1;
source = (unsigned char) id >> 0;
priority = (unsigned char) ((id >> 26) & 0x7);
if (canIdPF < 240) {
/* PDU1 format, the PS contains the destination address */
destination = canIdPS;
pgn = (((unsigned long)canIdDP) << 16) | (((unsigned long)canIdPF) << 8);
} else {
/* PDU2 format, the destination is implied global and the PGN is extended */
destination = 0xff;
pgn = (((unsigned long)canIdDP) << 16) | (((unsigned long)canIdPF) << 8) | (unsigned long)canIdPS;
}
};
MessageHeader(unsigned long PGN, unsigned char Priority, unsigned char Source, unsigned char Destination) {
// from NMEA2000 code base.
pgn = PGN;
priority = Priority;
source = Source;
destination = Destination;
unsigned char canIdPF = (unsigned char) (PGN >> 8);
if (canIdPF < 240) { // PDU1 format
if ( (PGN & 0xff) != 0 ) {
id = 0;
} else {
id = ( ((unsigned long)(priority & 0x7))<<26 | pgn<<8 | ((unsigned long)Destination)<<8 | (unsigned long)Source);
}
} else { // PDU2 format
id = ( ((unsigned long)(priority & 0x7))<<26 | pgn<<8 | (unsigned long)Source);
}
};
// |-| Priority
// PDU1 format DDSS
void print(Print *console, byte *buf, int len) {
console->print(":");
console->print(pgn);
console->print(",");
console->print(source);
console->print(",");
console->print(destination);
console->print(",");
console->print(priority);
console->print(",");
console->print(len);
console->print(",");
console->print(id,HEX);
for (int i = 0; i < len; i++) {
console->print(",");
console->print(buf[i],HEX);
}
console->println(";");
}
unsigned long id;
unsigned long pgn;
unsigned char priority;
unsigned char source;
unsigned char destination;
};
class SNMEA2000 {
public:
SNMEA2000(byte addr,
SNMEA2000DeviceInfo * devInfo,
const SNMEA2000ProductInfo * pinfo,
const SNMEA2000ConfigInfo * cinfo,
const unsigned long *tx,
const unsigned long *rx,
const uint8_t csPin,
Print * console = &Serial
):
deviceAddress{addr},
devInfo{devInfo},
productInfo{pinfo},
configInfo{cinfo},
txPGNList{tx},
rxPGNList{rx} ,
CAN{csPin},
console{console}
{
};
bool open(byte clockSet = MCP_8MHz);
void processMessages();
void dumpStatus() {
console->print(F("NMEA2000 Status open="));
console->print(canIsOpen);
console->print(F(" sent="));
console->print(messagesSent);
console->print(F(" recieved="));
console->print(messagesRecieved);
console->print(F(" packet errors="));
console->print(packetErrors);
console->print(F(" frame errors="));
console->println(frameErrors);
};
void setDiagnostics(bool enabled) {
diagnostics = enabled;
};
void sendMessage(MessageHeader *messageHeader, byte *message, int len);
//void sendFastPacket(MessageHeader *messageHeader, byte *message, int len, bool progmem=false);
void setSerialNumber(uint32_t serialNumber) {
devInfo->setSerialNumber(serialNumber);
};
void setDeviceAddress(unsigned char _deviceAddress) {
deviceAddress = _deviceAddress;
console->print("Device Address set to ");
console->println(deviceAddress);
};
unsigned char getAddress() { return deviceAddress; };
void startPacket(MessageHeader *messageHeader);
void finishPacket();
void startFastPacket(MessageHeader *messageHeader, int length);
void finishFastPacket();
void checkFastPacket();
void outputByte(byte opb);
void output3ByteInt(int32_t i);
void output2ByteUInt(uint16_t i);
void outputFixedString(const char * str, int maxLen, byte padding);
void outputVarString(const char * str, uint8_t strLen);
void output2ByteInt(uint16_t i);
void output2ByteDouble(double v, double p);
void output2ByteUDouble(double v, double p);
void output3ByteDouble(double v, double p);
void output3ByteUDouble(double v, double p);
void output4ByteDouble(double v, double p);
void output4ByteUDouble(double v, double p);
void setIsoRequestHandler(bool (*_isoRequestHandler)(unsigned long requestedPGN, MessageHeader *messageHeader, byte * buffer, int len)) {
isoRequestHandler = _isoRequestHandler;
};
static const byte broadcastAddress=0xff;
static const int16_t undefined2ByteDouble=0x7ffe;
static constexpr double n2kDoubleNA=-1000000000.0;
static const uint8_t n2kInt8NA=127;
private:
void handleISOAddressClaim(MessageHeader *messageHeader, byte * buffer, int len);
void claimAddress();
bool hasClaimedAddress();
void handleISORequest(MessageHeader *messageHeader, byte * buffer, int len);
void sendPGNLists(MessageHeader *requestMessageHeader);
void sendPGNList(MessageHeader *messageHeader, int listType, const unsigned long *pgnList);
void sendIsoAddressClaim();
void sendProductInformation(MessageHeader *requestMessageHeader);
void sendConfigurationInformation(MessageHeader *requestMessageHeader);
void sendIsoAcknowlegement(MessageHeader *requestMessageHeader, byte control, byte groupFunction);
bool clearRXFilter();
bool setupRXFilter();
int getPgmSize(const char *str, int maxLen);
void print_uint64_t(uint64_t num);
void print(tUnionDeviceInformation * devInfo) {
console->print(F(" UniqueNumber:"));
console->println(devInfo->unicNumberAndManCode&0x1fffff);
console->print(F(" ManufacturersCode:"));
console->println((devInfo->unicNumberAndManCode>>21)&0x7ff);
console->print(F(" deviceInstance:"));
console->println(devInfo->deviceInstance, DEC);
console->print(F(" deviceFunction:"));
console->println(devInfo->deviceFunction, DEC);
console->print(F(" deviceClass:"));
console->println((devInfo->deviceClass>>1)&0x7f, DEC);
console->print(F(" industryGroup:"));
console->println((devInfo->industryGroupAndSystemInstance>>4)&0x07, DEC);
console->print(F(" systemInstance:"));
console->println((devInfo->industryGroupAndSystemInstance>>4)&0x0f, DEC);
};
unsigned char deviceAddress;
SNMEA2000DeviceInfo * devInfo;
const SNMEA2000ProductInfo * productInfo;
const SNMEA2000ConfigInfo * configInfo;
const unsigned long *txPGNList;
const unsigned long *rxPGNList;
MCP_CAN CAN;
bool (*isoRequestHandler)(unsigned long requestedPGN, MessageHeader *messageHeader, byte * buffer, int len) = NULL;
unsigned long addressClaimStarted=0;
//output buffer and frames
MessageHeader *packetMessageHeader;
bool fastPacket;
bool rxFiltersSet = false;
bool diagnostics = false;
bool canIsOpen = false;
uint8_t fastPacketSequence;
int16_t fastPacketSent;
int16_t fastPacketLength;
byte buffer[8];
uint8_t frame;
uint8_t ob;
uint16_t messagesRecieved = 0;
uint16_t messagesSent = 0;
uint16_t packetErrors = 0;
uint16_t frameErrors = 0;
protected:
Print * console;
};
class PressureMonitor : public SNMEA2000 {
public:
PressureMonitor(byte addr,
SNMEA2000DeviceInfo * devInfo,
const SNMEA2000ProductInfo * pinfo,
const SNMEA2000ConfigInfo * cinfo,
const unsigned long *tx,
const unsigned long *rx,
const uint8_t csPin
): SNMEA2000{addr, devInfo, pinfo, cinfo, tx, rx, csPin} {};
/**
* @brief PGN 130310
*
* @param sid sequence ID
* @param waterTemperature in K
* @param outsideAirTemperature in K
* @param atmospheicPressure in Pascals
*/
void sendOutsideEnvironmentParameters(
byte sid=0,
double waterTemperature = SNMEA2000::n2kDoubleNA,
double outsideAirTemperature = SNMEA2000::n2kDoubleNA,
double atmospheicPressure=SNMEA2000::n2kDoubleNA
);
/**
* @brief PGN 130311L
*
* @param sid sequence ID
* @param atmosphericPressure in Pascals
* @param tempSource
* @param temperature in K
* @param humiditySource
* @param humidity in % RH
*/
void sendEnvironmentParameters(
byte sid=0,
double atmosphericPressure=SNMEA2000::n2kDoubleNA, // Atomspheric Pressure
byte tempSource=0,
double temperature=SNMEA2000::n2kDoubleNA, // temperature
byte humiditySource=0,
double humidity=SNMEA2000::n2kDoubleNA // humidity
);
/**
* @brief PGN 130313L
* @param sid
* @param humiditySource
* @param humidityInstance
* @param humidity in percent
*/
void sendHumidity(byte sid, byte humiditySource, byte humidityInstance, double humidity);
/**
* @brief PGN 130314L
* @param sid
* @param pressureSource
* @param pressureInstance
* @param pressure in Pascal
*/
void sendPressure(byte sid, byte pressureSource, byte pressureInstance, double pressure );
/**
* @brief PGN 130316L
* @param sid
* @param temperatureSource
* @param temperatureInstance
* @param temperature in K
*/
void sendTemperature(byte sid, byte temperatureSource, byte temperatureInstance, double temperature );
};
class EngineMonitor : public SNMEA2000 {
public:
EngineMonitor(byte addr,
SNMEA2000DeviceInfo * devInfo,
const SNMEA2000ProductInfo * pinfo,
const SNMEA2000ConfigInfo * cinfo,
const unsigned long *tx,
const unsigned long *rx,
const uint8_t csPin
): SNMEA2000{addr, devInfo, pinfo, cinfo, tx, rx, csPin} {};
/**
* RapidEngine Data - PGN 127488, standard packet
* enginInstance starting a 0
* engineSpeed in RPM
* engineBoostPressure in Pascal
* engineTrim in %
*/
void sendRapidEngineDataMessage(
byte engineInstance=0,
double engineSpeed=SNMEA2000::n2kDoubleNA, // RPM
double engineBoostPressure=SNMEA2000::n2kDoubleNA, // Pa
byte engineTiltTrim=SNMEA2000::n2kInt8NA // %
);
/**
* EngineDynamicParams - PGN 127489, fast packet
* engineInstance starting at 0
* engineHours in seconds
* engingCoolantTemperature in K
* alternator Volage in V
* status1 bitmap
* status2 bitmap
* engineOilPressure in Pascal
* engineOilPTemperature in K
* fuelRate in l/h
* engineCoolantPressure in Pascal
* engineFuelPressure in Pascal
* engineLoad in %
* engineTorque in %
*/
void sendEngineDynamicParamMessage(
byte engineInstance = 0,
double engineHours = SNMEA2000::n2kDoubleNA, // s
double engineCoolantTemperature = SNMEA2000::n2kDoubleNA, // K
double alternatorVoltage = SNMEA2000::n2kDoubleNA, // V
uint16_t status1 = 0,
uint16_t status2 = 0,
double engineOilPressure = SNMEA2000::n2kDoubleNA, // Pa
double engineOilPTemperature = SNMEA2000::n2kDoubleNA, // K
double fuelRate = SNMEA2000::n2kDoubleNA, // l/h
double engineCoolantPressure = SNMEA2000::n2kDoubleNA, // K
double engineFuelPressure = SNMEA2000::n2kDoubleNA, // Pa
byte engineLoad = SNMEA2000::n2kInt8NA, // %
byte engineTorque = SNMEA2000::n2kInt8NA // %
);
/**
* DC Battery Status PGN 127508
* batteryInstance starting a 0
* SID sequence id
* batterVoltage in volts
* batterTemperature in K
* batteryCurrent in A
*/
void sendDCBatterStatusMessage(
byte batteryInstance=0,
byte sid=0,
double batteryVoltage = SNMEA2000::n2kDoubleNA, // V
double batteryTemperature = SNMEA2000::n2kDoubleNA, // K
double batteryCurrent = SNMEA2000::n2kDoubleNA // A
);
/**
* Fluid Level PGN 127505
* type fuel=0,
* water=1,
* greywater=2,
* levelwell=3,
* oil=4,
* blackwater=5,
* petrol=6
* error=14
* unavailable=15
* instance unique tank instance
* level in %
* capacity in l
*/
void sendFluidLevelMessage(
byte type = 0,
byte instance = 0,
double level = SNMEA2000::n2kDoubleNA, // %
double capacity = SNMEA2000::n2kDoubleNA // l
);
/*
* Temperature PGN 130312
* sid sequence id
* intance instance starting at 0
* source
* sea=0,
* outside=1
* inside=2
* engineroom=3
* maincabin=4
* livewell=5
* baitwell=6
* fridge=7
* heating=8
* dewpoint=9
* apparent wind chill=10
* theoretical wind chill = 11
* heatindextemperature=12
* freezer=13
* exhaustgas=14
* actual in K
* requested in K
*/
void sendTemperatureMessage(
byte sid = 0,
byte instance = 0,
byte source = 0,
double actual = SNMEA2000::n2kDoubleNA, // K
double requested = SNMEA2000::n2kDoubleNA // K
);
};
// message type structures.
/*
typedef union EngineDynamicParamMessage {
byte buffer[26];
struct {
byte engineInstance = 0; //
uint16_t engineOilPressure = SNMEA2000::undefined2ByteUDouble; // in 1 = 100
uint16_t engineOilPTemperature = SNMEA2000::undefined2ByteUDouble; // in 1 = 0.1
uint16_t engineCoolantTemperature = SNMEA2000::undefined2ByteUDouble; // in 1 = 0.01
int16_t alternatorVoltage = SNMEA2000::undefined2ByteDouble; // in 1 = 0.01
int16_t fuelRate = SNMEA2000::undefined2ByteDouble; // in 1 = 0.1
uint32_t engineHours = SNMEA2000::undefined4ByteUDouble; // in 1 = 1
uint16_t engineCoolantPressure = SNMEA2000::undefined2ByteUDouble; // in 1 = 100
uint16_t engineFuelPressure = SNMEA2000::undefined2ByteUDouble; // in 1 = 1000
byte reserved1 = 0xff;
uint16_t status1;
uint16_t status2;
byte engineLoad;
byte engineTorque;
};
} EngineDynamicParamMessage;
typedef union DCBatStatusMessage {
byte buffer[8];
struct {
byte batteryInstance = 0;
uint16_t batteryVoltage = SNMEA2000::undefined2ByteDouble; // in 1 = 0.01
uint16_t batteryCurrent = SNMEA2000::undefined2ByteDouble; // in 1 = 0.1
uint16_t batteryTemperature = SNMEA2000::undefined2ByteDouble; // 1 = 0.01
byte SID = 1;
};
} DCBatStatusMessage;
typedef union FluidLevelMessage {
byte buffer[8];
struct {
byte typeInstance = 0; // hi nibble is type, low nibble is instance.
int16_t level = SNMEA2000::undefined2ByteDouble; // in 1 = 0.004
uint32_t capacity = SNMEA2000::undefined4ByteUDouble; // in 1 = 0.1
byte reserved = 0xff;
};
} FluidLevelMessage;
typedef union TemperatureMessage {
byte buffer[8];
struct {
byte SID = 0;
byte instance = 0;
byte source = 0;
uint16_t actual = SNMEA2000::undefined2ByteUDouble; // in 1 = 0.01
uint16_t requested = SNMEA2000::undefined2ByteUDouble; // in 1 = 0.01
byte reserved = 0xff;
};
} TemperatureMessage;
*/
#endif