OpenCPN Partial API docs
Loading...
Searching...
No Matches
comm_drv_n2k_serial.cpp
1/***************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: Implement comm_drv_n2k.h -- Nmea2000 serial driver.
5 * Author: David Register, Alec Leamas
6 *
7 ***************************************************************************
8 * Copyright (C) 2022 by David Register, Alec Leamas *
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 * This program is distributed in the hope that it will be useful, *
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
18 * GNU General Public License for more details. *
19 * *
20 * You should have received a copy of the GNU General Public License *
21 * along with this program; if not, write to the *
22 * Free Software Foundation, Inc., *
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
24 **************************************************************************/
25
26// For compilers that support precompilation, includes "wx.h".
27#include <wx/wxprec.h>
28
29#ifndef WX_PRECOMP
30#include <wx/wx.h>
31#endif // precompiled headers
32
33#include <vector>
34#include <mutex> // std::mutex
35#include <queue> // std::queue
36
37#include <wx/log.h>
38
39#include "model/comm_drv_n2k_serial.h"
42#include "model/logger.h"
44
45#include <N2kMsg.h>
46
47/* Copied from canboat Project
48 * https://github.com/canboat/canboat
49 *
50 * The following startup command reverse engineered from Actisense NMEAreader.
51 * It instructs the NGT1 to clear its PGN message TX list, thus it starts
52 * sending all PGNs that it knows about.
53 */
54static unsigned char NGT_STARTUP_SEQ[] = {
55 0x11, /* msg byte 1, meaning ? */
56 0x02, /* msg byte 2, meaning ? */
57 0x00 /* msg byte 3, meaning ? */
58};
59
60std::vector<unsigned char> BufferToActisenseFormat(tN2kMsg& msg);
61
62template <typename T>
64public:
65 size_t size() {
66 std::lock_guard<std::mutex> lock(m_mutex);
67 return m_queque.size();
68 }
69
70 bool empty() {
71 std::lock_guard<std::mutex> lock(m_mutex);
72 return m_queque.empty();
73 }
74
75 const T& front() {
76 std::lock_guard<std::mutex> lock(m_mutex);
77 return m_queque.front();
78 }
79
80 void push(const T& value) {
81 std::lock_guard<std::mutex> lock(m_mutex);
82 m_queque.push(value);
83 }
84
85 void pop() {
86 std::lock_guard<std::mutex> lock(m_mutex);
87 m_queque.pop();
88 }
89
90private:
91 std::queue<T> m_queque;
92 mutable std::mutex m_mutex;
93};
94
95template <class T>
96class circular_buffer {
97public:
98 explicit circular_buffer(size_t size)
99 : buf_(std::unique_ptr<T[]>(new T[size])), max_size_(size) {}
100
101 void reset();
102 size_t capacity() const;
103 size_t size() const;
104
105 bool empty() const {
106 // if head and tail are equal, we are empty
107 return (!full_ && (head_ == tail_));
108 }
109
110 bool full() const {
111 // If tail is ahead the head by 1, we are full
112 return full_;
113 }
114
115 void put(T item) {
116 std::lock_guard<std::mutex> lock(mutex_);
117 buf_[head_] = item;
118 if (full_) tail_ = (tail_ + 1) % max_size_;
119
120 head_ = (head_ + 1) % max_size_;
121
122 full_ = head_ == tail_;
123 }
124
125 T get() {
126 std::lock_guard<std::mutex> lock(mutex_);
127
128 if (empty()) return T();
129
130 // Read data and advance the tail (we now have a free space)
131 auto val = buf_[tail_];
132 full_ = false;
133 tail_ = (tail_ + 1) % max_size_;
134
135 return val;
136 }
137
138private:
139 std::mutex mutex_;
140 std::unique_ptr<T[]> buf_;
141 size_t head_ = 0;
142 size_t tail_ = 0;
143 const size_t max_size_;
144 bool full_ = 0;
145};
146
147class CommDriverN2KSerialEvent; // fwd
148
149class CommDriverN2KSerialThread : public wxThread {
150public:
152 const wxString& PortName,
153 const wxString& strBaudRate);
154
156 void* Entry();
157 bool SetOutMsg(const std::vector<unsigned char>& load);
158 void OnExit(void);
159 DriverStats GetStats() const;
160
161private:
162#ifndef __ANDROID__
163 serial::Serial m_serial;
164#endif
165 void ThreadMessage(const wxString& msg);
166 bool OpenComPortPhysical(const wxString& com_name, int baud_rate);
167 void CloseComPortPhysical();
168 size_t WriteComPortPhysical(std::vector<unsigned char> msg);
169 size_t WriteComPortPhysical(unsigned char* msg, size_t length);
170 void SetGatewayOperationMode(void);
171
172 CommDriverN2KSerial* m_pParentDriver;
173 wxString m_PortName;
174 wxString m_FullPortName;
175
176 unsigned char* put_ptr;
177 unsigned char* tak_ptr;
178
179 unsigned char* rx_buffer;
180
181 int m_baud;
182 int m_n_timeout;
183
185 DriverStats m_driver_stats;
186 mutable std::mutex m_stats_mutex;
187#ifdef __WXMSW__
188 HANDLE m_hSerialComm;
189 bool m_nl_found;
190#endif
191};
192
194wxDECLARE_EVENT(wxEVT_COMMDRIVER_N2K_SERIAL, CommDriverN2KSerialEvent);
195
196class CommDriverN2KSerialEvent : public wxEvent {
197public:
198 CommDriverN2KSerialEvent(wxEventType commandType = wxEVT_NULL, int id = 0)
199 : wxEvent(id, commandType) {};
201
202 // accessors
203 void SetPayload(std::shared_ptr<std::vector<unsigned char>> data) {
204 m_payload = data;
205 }
206 std::shared_ptr<std::vector<unsigned char>> GetPayload() { return m_payload; }
207
208 // required for sending with wxPostEvent()
209 wxEvent* Clone() const {
211 newevent->m_payload = this->m_payload;
212 return newevent;
213 };
214
215private:
216 std::shared_ptr<std::vector<unsigned char>> m_payload;
217};
218
219//========================================================================
220/* commdriverN2KSerial implementation
221 * */
222
223wxDEFINE_EVENT(wxEVT_COMMDRIVER_N2K_SERIAL, CommDriverN2KSerialEvent);
224
225CommDriverN2KSerial::CommDriverN2KSerial(const ConnectionParams* params,
226 DriverListener& listener)
227 : CommDriverN2K(params->GetStrippedDSPort()),
228 m_Thread_run_flag(-1),
229 m_params(*params),
230 m_bok(false),
231 m_portstring(params->GetDSPort()),
232 m_pSecondary_Thread(NULL),
233 m_listener(listener),
234 m_stats_timer(*this, 2s) {
235 m_BaudRate = wxString::Format("%i", params->Baudrate), SetSecThreadInActive();
236 m_manufacturers_code = 0;
237 m_got_mfg_code = false;
238 this->attributes["canAddress"] = std::string("-1");
239 this->attributes["userComment"] = params->UserComment.ToStdString();
240 this->attributes["ioDirection"] = std::string("IN/OUT");
241
242 // Prepare the wxEventHandler to accept events from the actual hardware thread
243 Bind(wxEVT_COMMDRIVER_N2K_SERIAL, &CommDriverN2KSerial::handle_N2K_SERIAL_RAW,
244 this);
245
246 // Dummy Driver Stats, may be polled before worker thread is active
247 m_driver_stats.driver_bus = NavAddr::Bus::N2000;
248 m_driver_stats.driver_iface = m_params.GetStrippedDSPort();
249 m_driver_stats.available = false;
250
251 Open();
252
253 wxMilliSleep(100);
254 GetMfgCode();
255
256 // Initialize the device clearing all rx/tx filterx
257 SendMgmtMsg(NGT_STARTUP_SEQ, sizeof(NGT_STARTUP_SEQ), 0x11, 0, NULL);
258
259#if 0
260 // Testing TX of Heartbeat
261 wxSleep(1);
262
263 tN2kMsg N2kMsg; // automatically sets destination 255
264 //SetHeartbeat(N2kMsg,2000,0);
265 //SetN2kPGN126993(N2kMsg, 2000, 0);
266 N2kMsg.SetPGN(126993L);
267 //N2kMsg.Priority=7;
268 N2kMsg.Source = 2;
269 N2kMsg.Destination = 133;
270 N2kMsg.Add2ByteUInt((uint16_t)(2000)); // Rate, msec
271
272 N2kMsg.AddByte(0); //Status
273 N2kMsg.AddByte(0xff); // Reserved
274 N2kMsg.Add4ByteUInt(0xffffffff); // Reserved
275
276 const std::vector<unsigned char> mv = BufferToActisenseFormat(N2kMsg);
277
278 size_t len = mv.size();
279
280 wxString comx = m_params.GetDSPort().AfterFirst(':');
281 std::string interface = comx.ToStdString();
282
283 N2kName source_name(1234);
284 auto source_address = std::make_shared<NavAddr2000>(interface, source_name);
285 auto dest_address = std::make_shared<NavAddr2000>(interface, N2kMsg.Destination);
286
287 auto message_to_send = std::make_shared<Nmea2000Msg>(126993L,
288 mv, source_address, 3);
289
290 for(size_t i=0; i< mv.size(); i++){
291 printf("%02X ", mv.at(i));
292 }
293 printf("\n\n");
294
295 SetTXPGN(126993);
296 wxSleep(1);
297
298 SendMessage(message_to_send, dest_address);
299
300 int yyp = 4;
301#endif
302}
303
304CommDriverN2KSerial::~CommDriverN2KSerial() { Close(); }
305
306DriverStats CommDriverN2KSerial::GetDriverStats() const {
307 if (m_pSecondary_Thread)
308 return m_pSecondary_Thread->GetStats();
309 else
310 return m_driver_stats;
311}
312
313bool CommDriverN2KSerial::Open() {
314 wxString comx;
315 comx = m_params.GetDSPort().AfterFirst(':'); // strip "Serial:"
316
317 comx =
318 comx.BeforeFirst(' '); // strip off any description provided by Windows
319
320#ifndef ANDROID
321 // Kick off the RX thread
322 SetSecondaryThread(new CommDriverN2KSerialThread(this, comx, m_BaudRate));
323 SetThreadRunFlag(1);
324 GetSecondaryThread()->Run();
325#endif
326
327 return true;
328}
329
330void CommDriverN2KSerial::Close() {
331 wxLogMessage(
332 wxString::Format(_T("Closing N2K Driver %s"), m_portstring.c_str()));
333
334 // Kill off the Secondary RX Thread if alive
335 if (m_pSecondary_Thread) {
336 if (m_bsec_thread_active) // Try to be sure thread object is still alive
337 {
338 wxLogMessage(_T("Stopping Secondary Thread"));
339
340 m_Thread_run_flag = 0;
341 int tsec = 10;
342 while ((m_Thread_run_flag >= 0) && (tsec--)) wxSleep(1);
343
344 wxString msg;
345 if (m_Thread_run_flag < 0)
346 msg.Printf(_T("Stopped in %d sec."), 10 - tsec);
347 else
348 msg.Printf(_T("Not Stopped after 10 sec."));
349 wxLogMessage(msg);
350 }
351
352 m_pSecondary_Thread = NULL;
353 m_bsec_thread_active = false;
354 }
355}
356
357bool CommDriverN2KSerial::SendMessage(std::shared_ptr<const NavMsg> msg,
358 std::shared_ptr<const NavAddr> addr) {
359#ifndef ANDROID
360
361 auto msg_n2k = std::dynamic_pointer_cast<const Nmea2000Msg>(msg);
362 std::vector<uint8_t> load = msg_n2k->payload;
363
364 uint64_t _pgn = msg_n2k->PGN.pgn;
365 auto destination_address = std::static_pointer_cast<const NavAddr2000>(addr);
366
367 tN2kMsg N2kMsg; // automatically sets destination 255
368 N2kMsg.SetPGN(_pgn);
369 N2kMsg.Priority = msg_n2k->priority;
370 if (destination_address) N2kMsg.Destination = destination_address->address;
371
372 for (size_t i = 0; i < load.size(); i++) N2kMsg.AddByte(load.at(i)); // data
373
374 const std::vector<uint8_t> mv = BufferToActisenseFormat(N2kMsg);
375
376 // printf("mv\n");
377 // for(size_t i=0; i< mv.size(); i++){
378 // printf("%02X ", mv.at(i));
379 // }
380 // printf("\n\n");
381
382 if (GetSecondaryThread()) {
383 if (IsSecThreadActive()) {
384 int retry = 10;
385 while (retry) {
386 if (GetSecondaryThread()->SetOutMsg(mv))
387 return true;
388 else
389 retry--;
390 }
391 return false; // could not send after several tries....
392 } else
393 return false;
394 }
395#endif
396 return true;
397}
398
399void CommDriverN2KSerial::ProcessManagementPacket(
400 std::vector<unsigned char>* payload) {
401 if (payload->at(2) != 0xF2) { // hearbeat
402 // printf(" pl ");
403 // for (unsigned int i = 0; i < payload->size(); i++)
404 // printf("%02X ", payload->at(i));
405 // printf("\n");
406 }
407
408 switch (payload->at(2)) {
409 case 0x47:
410 m_bmg47_resp = true;
411 break;
412 case 0x01:
413 m_bmg01_resp = true;
414 break;
415 case 0x4B:
416 m_bmg4B_resp = true;
417 break;
418 case 0x041: {
419 m_bmg41_resp = true;
420 if (payload->at(3) == 0x02) { // ASCII device_common_name
421 std::string device_common_name;
422 for (unsigned int i = 0; i < 32; i++) {
423 device_common_name += payload->at(i + 14);
424 }
425 device_common_name += '\0';
426 m_device_common_name = device_common_name;
427 }
428 break;
429 }
430 case 0x042: {
431 m_bmg42_resp = true;
432 unsigned char name[8];
433 for (unsigned int i = 0; i < 8; i++) name[i] = payload->at(i + 15);
434
435 memcpy((void*)&NAME, name, 8);
436 // Extract the manufacturers code
437 int* f1 = (int*)&NAME;
438 int f1d = *f1;
439 m_manufacturers_code = f1d >> 21;
440 break;
441 }
442
443 default:
444 break;
445 }
446}
447
448static uint64_t PayloadToName(const std::vector<unsigned char> payload) {
449 uint64_t name;
450 memcpy(&name, reinterpret_cast<const void*>(payload.data()), sizeof(name));
451 return name;
452}
453
454void CommDriverN2KSerial::handle_N2K_SERIAL_RAW(
456 auto p = event.GetPayload();
457
458 std::vector<unsigned char>* payload = p.get();
459
460 if (payload->at(0) == 0xA0) {
461 ProcessManagementPacket(payload);
462 return;
463 }
464
465 // extract PGN
466 uint64_t pgn = 0;
467 unsigned char* c = (unsigned char*)&pgn;
468 *c++ = payload->at(3);
469 *c++ = payload->at(4);
470 *c++ = payload->at(5);
471 // memcpy(&v, &data[3], 1);
472 // printf(" %ld\n", pgn);
473
474 auto name = PayloadToName(*payload);
475 auto msg =
476 std::make_shared<const Nmea2000Msg>(pgn, *payload, GetAddress(name));
477 auto msg_all =
478 std::make_shared<const Nmea2000Msg>(1, *payload, GetAddress(name));
479
480 m_listener.Notify(std::move(msg));
481 m_listener.Notify(std::move(msg_all));
482
483#if 0 // Debug output
484 size_t packetLength = (size_t)payload->at(1);
485 size_t vector_length = payload->size();
486
487
488 //if(pgn > 120000)
489 {
490 printf("Payload Length: %ld\n", vector_length);
491
492 printf("PGN: %ld\n", pgn);
493
494 for(size_t i=0; i< vector_length ; i++){
495 printf("%02X ", payload->at(i));
496 }
497 printf("\n\n");
498 }
499#endif
500}
501
502int CommDriverN2KSerial::GetMfgCode() {
503 unsigned char request_name[] = {0x42};
504 int ni = SendMgmtMsg(request_name, sizeof(request_name), 0x41, 2000,
505 &m_bmg42_resp);
506 if (ni) return ni; // Not responding, return error so upper retries.
507 m_got_mfg_code = true;
508 return 0;
509}
510
511int CommDriverN2KSerial::SendMgmtMsg(unsigned char* string, size_t string_size,
512 unsigned char cmd_code, int timeout_msec,
513 bool* response_flag) {
514#ifndef ANDROID
515 // Make a message
516 int byteSum = 0;
517 uint8_t CheckSum;
518 std::vector<unsigned char> msg;
519
520 msg.push_back(ESCAPE);
521 msg.push_back(STARTOFTEXT);
522 msg.push_back(0xA1);
523 byteSum += 0xA1;
524 msg.push_back(string_size); // payload length
525 byteSum += string_size;
526
527 for (unsigned int i = 0; i < string_size; i++) {
528 if (string[i] == ESCAPE) msg.push_back(string[i]);
529 msg.push_back(string[i]);
530 byteSum += string[i];
531 }
532
533 // checksum
534 byteSum %= 256;
535 CheckSum = (uint8_t)((byteSum == 0) ? 0 : (256 - byteSum));
536 msg.push_back(CheckSum);
537
538 msg.push_back(ESCAPE);
539 msg.push_back(ENDOFTEXT);
540
541 // send it out
542
543 if (response_flag) *response_flag = false; // prime the response detector
544
545 // Send the msg
546 bool bsent = false;
547 bool not_done = true;
548 int ntry_outer = 10;
549 while (not_done) {
550 if (GetSecondaryThread() && IsSecThreadActive()) {
551 int retry = 10;
552 while (retry) {
553 if (GetSecondaryThread()->SetOutMsg(msg)) {
554 bsent = true;
555 not_done = false;
556 break;
557 } else
558 retry--;
559 }
560 } else {
561 wxMilliSleep(100);
562 if (ntry_outer-- <= 0) not_done = false;
563 }
564 }
565
566 if (!bsent) return 1;
567
568 bool bOK = false;
569 if (timeout_msec) {
570 int timeout = timeout_msec;
571 while (timeout > 0) {
572 wxYieldIfNeeded();
573 wxMilliSleep(100);
574 if (response_flag) {
575 if (*response_flag) {
576 bOK = true;
577 break;
578 }
579 }
580 timeout -= 100;
581 }
582 } else
583 bOK = true;
584
585 if (!bOK) {
586 // printf( "***Err-1\n");
587 return 1;
588 }
589 // else
590 // printf("***OK-1 %d\n", timeout);
591#endif
592 return 0;
593}
594
595int CommDriverN2KSerial::SetTXPGN(int pgn) {
596 // Enable PGN message
597 unsigned char request_enable[] = {0x47, 0x00, 0x00, 0x00, // pgn
598 0x00, 0x01, 0xFF, 0xFF, 0xFF, 0xFF};
599
600 int PGN = 0;
601 unsigned char* c = (unsigned char*)&pgn;
602 request_enable[1] = c[0];
603 request_enable[2] = c[1];
604 request_enable[3] = c[2];
605
606 int aa = SendMgmtMsg(request_enable, sizeof(request_enable), 0x47, 2000,
607 &m_bmg47_resp);
608 if (aa) return 0; // Probably YDNU-02 device
609
610 // Commit message
611 unsigned char request_commit[] = {0x01};
612 int bb = SendMgmtMsg(request_commit, sizeof(request_commit), 0x01, 2000,
613 &m_bmg01_resp);
614
615 // Activate message
616 unsigned char request_activate[] = {0x4B};
617 int cc = SendMgmtMsg(request_activate, sizeof(request_activate), 0x4B, 2000,
618 &m_bmg4B_resp);
619
620 return 0;
621}
622void CommDriverN2KSerial::AddTxPGN(int pgn) {
623 auto it = std::find(pgn_tx_list.begin(), pgn_tx_list.end(), pgn);
624 if (it != pgn_tx_list.end())
625 return;
626 else {
627 SetTXPGN(pgn);
628 pgn_tx_list.push_back(pgn);
629 }
630}
631
632#ifndef __ANDROID__
633
640// Commonly used raw format is actually inherited from an old paketizing format:
641// <10><02><application data><CRC (1)><10><03>
642
643// Actisense application data, from NGT-1 to PC
644// <data code=93><length (1)><priority (1)><PGN (3)><destination(1)><source
645// (1)><time (4)><len (1)><data (len)>
646
647// As applied to a real application data element, after extraction from packet
648// format: 93 13 02 01 F8 01 FF 01 76 C2 52 00 08 08 70 EB 14 E8 8E 52 D2 BB 10
649
650// length (1): 0x13
651// priority (1); 0x02
652// PGN (3): 0x01 0xF8 0x01
653// destination(1): 0xFF
654// source (1): 0x01
655// time (4): 0x76 0xC2 0x52 0x00
656// len (1): 0x08
657// data (len): 08 70 EB 14 E8 8E 52 D2
658// packet CRC: 0xBB
659
660#define DS_RX_BUFFER_SIZE 4096
661
662CommDriverN2KSerialThread::CommDriverN2KSerialThread(
663 CommDriverN2KSerial* Launcher, const wxString& PortName,
664 const wxString& strBaudRate) {
665 m_pParentDriver = Launcher; // This thread's immediate "parent"
666
667 m_PortName = PortName;
668 m_FullPortName = _T("Serial:") + PortName;
669
670 rx_buffer = new unsigned char[DS_RX_BUFFER_SIZE + 1];
671
672 put_ptr = rx_buffer; // local circular queue
673 tak_ptr = rx_buffer;
674
675 m_baud = 9600; // default
676 long lbaud;
677 if (strBaudRate.ToLong(&lbaud)) m_baud = (int)lbaud;
678 {
679 std::lock_guard lock(m_stats_mutex);
680 m_driver_stats.driver_bus = NavAddr::Bus::N2000;
681 m_driver_stats.driver_iface = m_pParentDriver->m_params.GetStrippedDSPort();
682 m_driver_stats.available = false;
683 }
684
685 Create();
686}
687
688CommDriverN2KSerialThread::~CommDriverN2KSerialThread(void) {
689 delete[] rx_buffer;
690}
691
692void CommDriverN2KSerialThread::OnExit(void) {}
693
694DriverStats CommDriverN2KSerialThread::GetStats() const {
695 std::lock_guard lock(m_stats_mutex);
696 return m_driver_stats;
697}
698
699bool CommDriverN2KSerialThread::OpenComPortPhysical(const wxString& com_name,
700 int baud_rate) {
701 try {
702 m_serial.setPort(com_name.ToStdString());
703 m_serial.setBaudrate(baud_rate);
704 m_serial.open();
705 m_serial.setTimeout(250, 250, 0, 250, 0);
706 } catch (std::exception&) {
707 // std::cerr << "Unhandled Exception while opening serial port: " <<
708 // e.what() << std::endl;
709 }
710 return m_serial.isOpen();
711}
712
713void CommDriverN2KSerialThread::CloseComPortPhysical() {
714 try {
715 m_serial.close();
716 } catch (std::exception&) {
717 // std::cerr << "Unhandled Exception while closing serial port: " <<
718 // e.what() << std::endl;
719 }
720 std::lock_guard lock(m_stats_mutex);
721 m_driver_stats.available = false;
722}
723
724void CommDriverN2KSerialThread::SetGatewayOperationMode(void) {
725 // For YDNU-02 device
726 // From Device User Manual
727 // Set the mode to "N2K"
728 unsigned char config_string[] = {0x10, 0x02, 0xA1, 0x03, 0x11,
729 0x02, 0x00, 0x49, 0x10, 0x03};
730 // std::vector<byte>writeBuffer {DLE,STX,NGT_TX_CMD,0x03,0x11,0x02,0x00,0x49,
731 // DLE,ETX};
732
733 WriteComPortPhysical(config_string, 10);
734}
735
736void CommDriverN2KSerialThread::ThreadMessage(const wxString& msg) {
737 // Signal the main program thread
738 // OCPN_ThreadMessageEvent event(wxEVT_OCPN_THREADMSG, 0);
739 // event.SetSString(std::string(msg.mb_str()));
740 // if (gFrame) gFrame->GetEventHandler()->AddPendingEvent(event);
741}
742
743size_t CommDriverN2KSerialThread::WriteComPortPhysical(
744 std::vector<unsigned char> msg) {
745 return WriteComPortPhysical(msg.data(), msg.size());
746}
747
748size_t CommDriverN2KSerialThread::WriteComPortPhysical(unsigned char* msg,
749 size_t length) {
750 if (!m_serial.isOpen()) return 0;
751 try {
752 size_t status = m_serial.write((uint8_t*)msg, length);
753 if (status) m_serial.flushOutput();
754 return status;
755 } catch (std::exception& e) {
756 DEBUG_LOG << "Unhandled Exception while writing to serial port: "
757 << e.what();
758 }
759 return 0;
760}
761
762bool CommDriverN2KSerialThread::SetOutMsg(
763 const std::vector<unsigned char>& msg) {
764 if (out_que.size() < OUT_QUEUE_LENGTH) {
765 out_que.push(msg);
766 return true;
767 }
768 return false;
769}
770
771#ifndef __WXMSW__
772void* CommDriverN2KSerialThread::Entry() {
773 bool not_done = true;
774 bool nl_found = false;
775 wxString msg;
776 uint8_t rdata[2000];
777 circular_buffer<uint8_t> circle(DS_RX_BUFFER_SIZE);
778 int ib = 0;
779
780 // Request the com port from the comm manager
781 if (!OpenComPortPhysical(m_PortName, m_baud)) {
782 wxString msg(_T("NMEA input device open failed: "));
783 msg.Append(m_PortName);
784 ThreadMessage(msg);
785 std::lock_guard lock(m_stats_mutex);
786 m_driver_stats.available = false;
787
788 // goto thread_exit; // This means we will not be trying to connect = The
789 // device must be connected when the thread is created. Does not seem to be
790 // needed/what we want as the reconnection logic is able to pick it up
791 // whenever it actually appears (Of course given it appears with the
792 // expected device name).
793 } else {
794 wxMilliSleep(100);
795 std::lock_guard lock(m_stats_mutex);
796 m_driver_stats.available = true;
797 SetGatewayOperationMode();
798 }
799
800 m_pParentDriver->SetSecThreadActive(); // I am alive
801
802 // The main loop
803 static size_t retries = 0;
804
805 bool bInMsg = false;
806 bool bGotESC = false;
807 bool bGotSOT = false;
808
809 while ((not_done) && (m_pParentDriver->m_Thread_run_flag > 0)) {
810 if (TestDestroy()) not_done = false; // smooth exit
811
812 uint8_t next_byte = 0;
813 int newdata = 0;
814 if (m_serial.isOpen()) {
815 try {
816 newdata = m_serial.read(rdata, 1000);
817 } catch (std::exception& e) {
818 // std::cerr << "Serial read exception: " << e.what() << std::endl;
819 std::lock_guard lock(m_stats_mutex);
820 m_driver_stats.error_count++;
821
822 if (10 < retries++) {
823 // We timed out waiting for the next character 10 times, let's close
824 // the port so that the reconnection logic kicks in and tries to fix
825 // our connection.
826 CloseComPortPhysical();
827 retries = 0;
828 }
829 }
830 } else {
831 // Reconnection logic. Let's try to reopen the port while waiting longer
832 // every time (until we simply keep trying every 2.5 seconds)
833 // std::cerr << "Serial port seems closed." << std::endl;
834 wxMilliSleep(250 * retries);
835 CloseComPortPhysical();
836 if (OpenComPortPhysical(m_PortName, m_baud)) {
837 SetGatewayOperationMode();
838 std::lock_guard lock(m_stats_mutex);
839 m_driver_stats.available = true;
840 retries = 0;
841 } else if (retries < 10) {
842 std::lock_guard lock(m_stats_mutex);
843 m_driver_stats.available = false;
844 retries++;
845 }
846 }
847
848 if (newdata > 0) {
849 std::lock_guard lock(m_stats_mutex);
850 m_driver_stats.rx_count += newdata;
851
852 for (int i = 0; i < newdata; i++) {
853 circle.put(rdata[i]);
854 }
855 }
856
857 while (!circle.empty()) {
858 if (ib >= DS_RX_BUFFER_SIZE) ib = 0;
859 uint8_t next_byte = circle.get();
860
861 if (bInMsg) {
862 if (bGotESC) {
863 if (ESCAPE == next_byte) {
864 rx_buffer[ib++] = next_byte;
865 bGotESC = false;
866 }
867 }
868
869 if (bGotESC && (ENDOFTEXT == next_byte)) {
870 // Process packet
871 // Copy the message into a std::vector
872
873 auto buffer = std::make_shared<std::vector<unsigned char>>(
874 rx_buffer, rx_buffer + ib);
875 std::vector<unsigned char>* vec = buffer.get();
876
877 ib = 0;
878 // tak_ptr = tptr;
879 bInMsg = false;
880 bGotESC = false;
881
882 // printf("raw ");
883 // for (unsigned int i = 0; i < vec->size(); i++)
884 // printf("%02X ", vec->at(i));
885 // printf("\n");
886
887 // Message is finished
888 // Send the captured raw data vector pointer to the thread's "parent"
889 // thereby releasing the thread for further data capture
890 CommDriverN2KSerialEvent Nevent(wxEVT_COMMDRIVER_N2K_SERIAL, 0);
891 Nevent.SetPayload(buffer);
892 m_pParentDriver->AddPendingEvent(Nevent);
893
894 } else {
895 bGotESC = (next_byte == ESCAPE);
896
897 if (!bGotESC) {
898 rx_buffer[ib++] = next_byte;
899 }
900 }
901 }
902
903 else {
904 if (STARTOFTEXT == next_byte) {
905 bGotSOT = false;
906 if (bGotESC) {
907 bGotSOT = true;
908 }
909 } else {
910 bGotESC = (next_byte == ESCAPE);
911 if (bGotSOT) {
912 bGotSOT = false;
913 bInMsg = true;
914
915 rx_buffer[ib++] = next_byte;
916 }
917 }
918 }
919 } // if newdata > 0
920
921 // Check for any pending output message
922#if 1
923 bool b_qdata = !out_que.empty();
924
925 while (b_qdata) {
926 // Take a copy of message
927 std::vector<unsigned char> qmsg = out_que.front();
928 out_que.pop();
929
930 if (static_cast<size_t>(-1) == WriteComPortPhysical(qmsg) &&
931 10 < retries++) {
932 // We failed to write the port 10 times, let's close the port so that
933 // the reconnection logic kicks in and tries to fix our connection.
934 retries = 0;
935 CloseComPortPhysical();
936 }
937
938 b_qdata = !out_que.empty();
939 } // while b_qdata
940
941#endif
942 } // while ((not_done)
943
944 // thread_exit:
945 CloseComPortPhysical();
946 m_pParentDriver->SetSecThreadInActive(); // I am dead
947 m_pParentDriver->m_Thread_run_flag = -1;
948
949 std::lock_guard lock(m_stats_mutex);
950 m_driver_stats.available = false;
951
952 return 0;
953}
954
955#else
956void* CommDriverN2KSerialThread::Entry() {
957 bool not_done = true;
958 bool nl_found = false;
959 wxString msg;
960 circular_buffer<uint8_t> circle(DS_RX_BUFFER_SIZE);
961
962 // Request the com port from the comm manager
963 if (!OpenComPortPhysical(m_PortName, m_baud)) {
964 wxString msg(_T("NMEA input device open failed: "));
965 msg.Append(m_PortName);
966 ThreadMessage(msg);
967 // goto thread_exit; // This means we will not be trying to connect = The
968 // device must be connected when the thread is created. Does not seem to be
969 // needed/what we want as the reconnection logic is able to pick it up
970 // whenever it actually appears (Of course given it appears with the
971 // expected device name).
972 } else {
973 SetGatewayOperationMode();
974 }
975
976 m_pParentDriver->SetSecThreadActive(); // I am alive
977
978 // The main loop
979 static size_t retries = 0;
980
981 bool bInMsg = false;
982 bool bGotESC = false;
983 bool bGotSOT = false;
984
985 while ((not_done) && (m_pParentDriver->m_Thread_run_flag > 0)) {
986 if (TestDestroy()) not_done = false; // smooth exit
987
988 uint8_t next_byte = 0;
989 int newdata = -1;
990 uint8_t rdata[2000];
991
992 if (m_serial.isOpen()) {
993 try {
994 newdata = m_serial.read(rdata, 200);
995 } catch (std::exception& e) {
996 // std::cerr << "Serial read exception: " << e.what() << std::endl;
997 if (10 < retries++) {
998 // We timed out waiting for the next character 10 times, let's close
999 // the port so that the reconnection logic kicks in and tries to fix
1000 // our connection.
1001 CloseComPortPhysical();
1002 retries = 0;
1003 }
1004 }
1005 } else {
1006 // Reconnection logic. Let's try to reopen the port while waiting longer
1007 // every time (until we simply keep trying every 2.5 seconds)
1008 // std::cerr << "Serial port seems closed." << std::endl;
1009 wxMilliSleep(250 * retries);
1010 CloseComPortPhysical();
1011 if (OpenComPortPhysical(m_PortName, m_baud)) {
1012 SetGatewayOperationMode();
1013 retries = 0;
1014 } else if (retries < 10)
1015 retries++;
1016 }
1017
1018 if (newdata > 0) {
1019 for (int i = 0; i < newdata; i++) {
1020 circle.put(rdata[i]);
1021 }
1022 }
1023
1024 while (!circle.empty()) {
1025 uint8_t next_byte = circle.get();
1026
1027 if (1) {
1028 if (bInMsg) {
1029 if (bGotESC) {
1030 if (ESCAPE == next_byte) {
1031 *put_ptr++ = next_byte;
1032 if ((put_ptr - rx_buffer) > DS_RX_BUFFER_SIZE)
1033 put_ptr = rx_buffer;
1034 bGotESC = false;
1035 } else if (ENDOFTEXT == next_byte) {
1036 // Process packet
1037 // Copy the message into a std::vector
1038
1039 auto buffer = std::make_shared<std::vector<unsigned char>>();
1040 std::vector<unsigned char>* vec = buffer.get();
1041
1042 unsigned char* tptr;
1043 tptr = tak_ptr;
1044
1045 while ((tptr != put_ptr)) {
1046 vec->push_back(*tptr++);
1047 if ((tptr - rx_buffer) > DS_RX_BUFFER_SIZE) tptr = rx_buffer;
1048 }
1049
1050 tak_ptr = tptr;
1051 bInMsg = false;
1052 bGotESC = false;
1053
1054 // Message is finished
1055 // Send the captured raw data vector pointer to the thread's
1056 // "parent"
1057 // thereby releasing the thread for further data capture
1058 CommDriverN2KSerialEvent Nevent(wxEVT_COMMDRIVER_N2K_SERIAL, 0);
1059 Nevent.SetPayload(buffer);
1060 m_pParentDriver->AddPendingEvent(Nevent);
1061 } else if (next_byte == STARTOFTEXT) {
1062 put_ptr = rx_buffer;
1063 bGotESC = false;
1064 } else {
1065 put_ptr = rx_buffer;
1066 bInMsg = false;
1067 bGotESC = false;
1068 }
1069
1070 } else {
1071 bGotESC = (next_byte == ESCAPE);
1072
1073 if (!bGotESC) {
1074 *put_ptr++ = next_byte;
1075 if ((put_ptr - rx_buffer) > DS_RX_BUFFER_SIZE)
1076 put_ptr = rx_buffer;
1077 }
1078 }
1079 }
1080
1081 else {
1082 if (STARTOFTEXT == next_byte) {
1083 bGotSOT = false;
1084 if (bGotESC) {
1085 bGotSOT = true;
1086 }
1087 } else {
1088 bGotESC = (next_byte == ESCAPE);
1089 if (bGotSOT) {
1090 bGotSOT = false;
1091 bInMsg = true;
1092
1093 *put_ptr++ = next_byte;
1094 if ((put_ptr - rx_buffer) > DS_RX_BUFFER_SIZE)
1095 put_ptr = rx_buffer;
1096 }
1097 }
1098 }
1099 } // if newdata > 0
1100 } // while
1101
1102 // Check for any pending output message
1103 bool b_qdata = !out_que.empty();
1104
1105 while (b_qdata) {
1106 // Take a copy of message
1107 std::vector<unsigned char> qmsg = out_que.front();
1108 out_que.pop();
1109
1110 if (static_cast<size_t>(-1) == WriteComPortPhysical(qmsg) &&
1111 10 < retries++) {
1112 // We failed to write the port 10 times, let's close the port so that
1113 // the reconnection logic kicks in and tries to fix our connection.
1114 retries = 0;
1115 CloseComPortPhysical();
1116 }
1117
1118 b_qdata = !out_que.empty();
1119 } // while b_qdata
1120 } // while ((not_done)
1121
1122 // thread_exit:
1123 CloseComPortPhysical();
1124 m_pParentDriver->SetSecThreadInActive(); // I am dead
1125 m_pParentDriver->m_Thread_run_flag = -1;
1126
1127 return 0;
1128}
1129
1130#endif // wxmsw Entry()
1131
1132#endif // Android
1133
1134//*****************************************************************************
1135// Actisense Format:
1136// <10><02><93><length (1)><priority (1)><PGN (3)><destination (1)><source
1137// (1)><time (4)><len (1)><data (len)><CRC (1)><10><03>
1138#define MaxActisenseMsgBuf 400
1139#define MsgTypeN2kTX 0x94
1140
1141void AddByteEscapedToBuf(unsigned char byteToAdd, uint8_t& idx,
1142 unsigned char* buf, int& byteSum);
1143
1144std::vector<unsigned char> BufferToActisenseFormat(tN2kMsg& msg) {
1145 unsigned long _PGN = msg.PGN;
1146 uint8_t msgIdx = 0;
1147 int byteSum = 0;
1148 uint8_t CheckSum;
1149 unsigned char ActisenseMsgBuf[MaxActisenseMsgBuf];
1150
1151 ActisenseMsgBuf[msgIdx++] = ESCAPE;
1152 ActisenseMsgBuf[msgIdx++] = STARTOFTEXT;
1153 AddByteEscapedToBuf(MsgTypeN2kTX, msgIdx, ActisenseMsgBuf, byteSum);
1154 AddByteEscapedToBuf(msg.DataLen + 6, msgIdx, ActisenseMsgBuf,
1155 byteSum); // length does not include escaped chars
1156
1157 AddByteEscapedToBuf(msg.Priority, msgIdx, ActisenseMsgBuf, byteSum);
1158 AddByteEscapedToBuf(_PGN & 0xff, msgIdx, ActisenseMsgBuf, byteSum);
1159 _PGN >>= 8;
1160 AddByteEscapedToBuf(_PGN & 0xff, msgIdx, ActisenseMsgBuf, byteSum);
1161 _PGN >>= 8;
1162 AddByteEscapedToBuf(_PGN & 0xff, msgIdx, ActisenseMsgBuf, byteSum);
1163 AddByteEscapedToBuf(msg.Destination, msgIdx, ActisenseMsgBuf, byteSum);
1164
1165#if 0
1166 // For TX through Actisense compatible gateway, we skip "source" byte and msg time fields
1167 // Source
1168 AddByteEscapedToBuf(msg.Source,msgIdx,ActisenseMsgBuf,byteSum);
1169 // Time
1170 int _MsgTime = 0;
1171 AddByteEscapedToBuf(_MsgTime & 0xff,msgIdx,ActisenseMsgBuf,byteSum); _MsgTime>>=8;
1172 AddByteEscapedToBuf(_MsgTime & 0xff,msgIdx,ActisenseMsgBuf,byteSum); _MsgTime>>=8;
1173 AddByteEscapedToBuf(_MsgTime & 0xff,msgIdx,ActisenseMsgBuf,byteSum); _MsgTime>>=8;
1174 AddByteEscapedToBuf(_MsgTime & 0xff,msgIdx,ActisenseMsgBuf,byteSum);
1175
1176#endif
1177
1178 AddByteEscapedToBuf(msg.DataLen, msgIdx, ActisenseMsgBuf, byteSum);
1179
1180 for (int i = 0; i < msg.DataLen; i++)
1181 AddByteEscapedToBuf(msg.Data[i], msgIdx, ActisenseMsgBuf, byteSum);
1182 byteSum %= 256;
1183
1184 CheckSum = (uint8_t)((byteSum == 0) ? 0 : (256 - byteSum));
1185 ActisenseMsgBuf[msgIdx++] = CheckSum;
1186 if (CheckSum == ESCAPE) ActisenseMsgBuf[msgIdx++] = CheckSum;
1187
1188 ActisenseMsgBuf[msgIdx++] = ESCAPE;
1189 ActisenseMsgBuf[msgIdx++] = ENDOFTEXT;
1190
1191 std::vector<unsigned char> rv;
1192 for (unsigned int i = 0; i < msgIdx; i++) rv.push_back(ActisenseMsgBuf[i]);
1193
1194 return rv;
1195}
Interface implemented by transport layer and possible other parties like test code which should handl...
Definition comm_driver.h:48
virtual void Notify(std::shared_ptr< const NavMsg > message)=0
Handle a received message.
Class that provides a portable serial port interface.
Definition serial.h:147
size_t read(uint8_t *buffer, size_t size)
Read a given amount of bytes from the serial port into a given buffer.
void setPort(const std::string &port)
Sets the serial port identifier.
size_t write(const uint8_t *data, size_t size)
Write a string to the serial port.
void setBaudrate(uint32_t baudrate)
Sets the baudrate for the serial port.
void close()
Closes the serial port.
void setTimeout(Timeout &timeout)
Sets the timeout for reads and writes using the Timeout struct.
void flushOutput()
Flush only the output buffer.
void open()
Opens the serial port as long as the port is set and the port isn't already open.
bool isOpen() const
Gets the open status of the serial port.
Driver registration container, a singleton.
Communication statistics infrastructure.
Raw messages layer, supports sending and recieving navmsg messages.
Enhanced logging interface on top of wx/log.h.
Driver statistics report.
unsigned rx_count
Number of bytes received since program start.
unsigned error_count
Number of detected errors since program start.
N2k uses CAN which defines the basic properties of messages.
Definition comm_navmsg.h:66