41#include <wx/msw/winundef.h>
47#include <netinet/tcp.h>
55#include <wx/tokenzr.h>
56#include <wx/datetime.h>
61#include <wx/chartype.h>
63#include <wx/sckaddr.h>
71#define N_DOG_TIMEOUT 8
73using namespace std::literals::chrono_literals;
75static const int kNotFound = -1;
80 void SetMrqAddr(
unsigned int addr) {
81 m_mrq.imr_multiaddr.s_addr = addr;
82 m_mrq.imr_interface.s_addr = INADDR_ANY;
88 : priority(
'\0'), source(
'\0'), destination(
'\0'), pgn(-1) {};
98 : wxEvent(
id, commandType) {};
102 void SetPayload(std::shared_ptr<std::vector<unsigned char>> data) {
105 std::shared_ptr<std::vector<unsigned char>> GetPayload() {
return m_payload; }
108 wxEvent* Clone()
const {
110 newevent->m_payload = this->m_payload;
115 std::shared_ptr<std::vector<unsigned char>> m_payload;
118static uint64_t PayloadToName(
const std::vector<unsigned char> payload) {
120 memcpy(&name,
reinterpret_cast<const void*
>(payload.data()),
sizeof(name));
128#define TIMER_SOCKET_N2KNET 7339
131EVT_TIMER(TIMER_SOCKET_N2KNET, CommDriverN2KNet::OnTimerSocket)
133EVT_SOCKET(DS_SERVERSOCKET_ID, CommDriverN2KNet::OnServerSocketEvent)
134EVT_TIMER(TIMER_SOCKET_N2KNET + 1, CommDriverN2KNet::OnSocketReadWatchdogTimer)
143 m_listener(listener),
144 m_stats_timer(*this, 2s),
145 m_net_port(wxString::Format("%i", params->NetworkPort)),
146 m_net_protocol(params->NetProtocol),
149 m_socket_server(NULL),
150 m_is_multicast(false),
152 m_portstring(params->GetDSPort()),
153 m_io_select(params->IOSelect),
154 m_connection_type(params->Type),
156 m_circle(RX_BUFFER_SIZE_NET),
157 m_TX_available(false) {
158 m_addr.Hostname(params->NetworkAddress);
159 m_addr.Service(params->NetworkPort);
161 m_driver_stats.driver_bus = NavAddr::Bus::N2000;
162 m_driver_stats.driver_iface = params->GetStrippedDSPort();
164 m_socket_timer.SetOwner(
this, TIMER_SOCKET_N2KNET);
165 m_socketread_watchdog_timer.SetOwner(
this, TIMER_SOCKET_N2KNET + 1);
166 this->attributes[
"netAddress"] = params->NetworkAddress.ToStdString();
168 sprintf(port_char,
"%d", params->NetworkPort);
169 this->attributes[
"netPort"] = std::string(port_char);
170 this->attributes[
"userComment"] = params->UserComment.ToStdString();
174 Bind(wxEVT_COMMDRIVER_N2K_NET, &CommDriverN2KNet::handle_N2K_MSG,
this);
176 m_prodinfo_timer.Connect(
177 wxEVT_TIMER, wxTimerEventHandler(CommDriverN2KNet::OnProdInfoTimer), NULL,
185 rx_buffer =
new unsigned char[RX_BUFFER_SIZE_NET + 1];
189 m_n2k_format = N2KFormat_YD_RAW;
192 resume_listener.Init(SystemEvents::GetInstance().evt_resume,
198CommDriverN2KNet::~CommDriverN2KNet() {
199 delete m_mrq_container;
206 std::string Model_ID;
210std::unordered_map<uint8_t, product_info> prod_info_map;
212bool CommDriverN2KNet::HandleMgntMsg(uint64_t pgn,
213 std::vector<unsigned char>& payload) {
215 auto name = PayloadToName(payload);
217 std::make_shared<const Nmea2000Msg>(pgn, payload, GetAddress(name));
219 bool b_handled =
false;
222 uint8_t src_addr = payload.at(7);
223 if (src_addr == 75)
return false;
225 pr_info.Model_ID = std::string((
char*)&payload.data()[17], 32);
226 pr_info.RT_flag = m_TX_flag;
228 prod_info_map[src_addr] = pr_info;
233 uint8_t src_addr = payload.at(7);
243void CommDriverN2KNet::OnProdInfoTimer(wxTimerEvent& ev) {
245 bool b_found =
false;
246 for (
const auto& [key, value] : prod_info_map) {
247 auto prod_info = value;
248 if (prod_info.Model_ID.find(
"YDEN") != std::string::npos) {
253 if (prod_info.RT_flag ==
'T') b_found =
true;
258 if (b_found) m_TX_available =
true;
259 prod_info_map.clear();
263 auto p =
event.GetPayload();
264 std::vector<unsigned char>* payload = p.get();
268 unsigned char* c = (
unsigned char*)&pgn;
269 *c++ = payload->at(3);
270 *c++ = payload->at(4);
271 *c++ = payload->at(5);
275 auto name = PayloadToName(*payload);
277 std::make_shared<const Nmea2000Msg>(pgn, *payload, GetAddress(name));
278 m_driver_stats.
rx_count += payload->size();
279 m_listener.
Notify(std::move(msg));
282void CommDriverN2KNet::Open() {
284#if wxCHECK_VERSION(3, 0, 0)
286 ((
struct sockaddr_in*)GetAddr().GetAddressData())->sin_addr.s_addr;
289 ((
struct sockaddr_in*)GetAddr().GetAddress()->m_addr)->sin_addr.s_addr;
292 unsigned int addr = inet_addr(GetAddr().IPAddress().mb_str());
295 switch (m_net_protocol) {
297 OpenNetworkTCP(addr);
301 OpenNetworkUDP(addr);
310void CommDriverN2KNet::OpenNetworkUDP(
unsigned int addr) {
311 if (GetPortType() != DS_TYPE_OUTPUT) {
314 wxIPV4address conn_addr;
315 conn_addr.Service(GetNetPort());
316 conn_addr.AnyAddress();
318 new wxDatagramSocket(conn_addr, wxSOCKET_NOWAIT | wxSOCKET_REUSEADDR));
321 if ((ntohl(addr) & 0xf0000000) == 0xe0000000) {
323 m_mrq_container->SetMrqAddr(addr);
324 GetSock()->SetOption(IPPROTO_IP, IP_ADD_MEMBERSHIP,
325 &m_mrq_container->m_mrq,
326 sizeof(m_mrq_container->m_mrq));
329 GetSock()->SetEventHandler(*
this, DS_SOCKET_ID);
331 GetSock()->SetNotify(wxSOCKET_CONNECTION_FLAG | wxSOCKET_INPUT_FLAG |
333 GetSock()->Notify(TRUE);
334 GetSock()->SetTimeout(1);
335 m_driver_stats.available =
true;
339 if (GetPortType() != DS_TYPE_INPUT) {
340 wxIPV4address tconn_addr;
341 tconn_addr.Service(0);
342 tconn_addr.AnyAddress();
344 new wxDatagramSocket(tconn_addr, wxSOCKET_NOWAIT | wxSOCKET_REUSEADDR));
349 if ((!GetMulticast()) && (GetAddr().IPAddress().EndsWith(
"255"))) {
350 int broadcastEnable = 1;
351 bool bam = GetTSock()->SetOption(
352 SOL_SOCKET, SO_BROADCAST, &broadcastEnable,
sizeof(broadcastEnable));
354 m_driver_stats.available =
true;
358 SetConnectTime(wxDateTime::Now());
361void CommDriverN2KNet::OpenNetworkTCP(
unsigned int addr) {
362 int isServer = ((addr == INADDR_ANY) ? 1 : 0);
363 wxLogMessage(wxString::Format(
"Opening TCP Server %d", isServer));
366 SetSockServer(
new wxSocketServer(GetAddr(), wxSOCKET_REUSEADDR));
368 SetSock(
new wxSocketClient());
372 GetSockServer()->SetEventHandler(*
this, DS_SERVERSOCKET_ID);
373 GetSockServer()->SetNotify(wxSOCKET_CONNECTION_FLAG);
374 GetSockServer()->Notify(TRUE);
375 GetSockServer()->SetTimeout(1);
377 GetSock()->SetEventHandler(*
this, DS_SOCKET_ID);
378 int notify_flags = (wxSOCKET_CONNECTION_FLAG | wxSOCKET_LOST_FLAG);
379 if (GetPortType() != DS_TYPE_INPUT) notify_flags |= wxSOCKET_OUTPUT_FLAG;
380 if (GetPortType() != DS_TYPE_OUTPUT) notify_flags |= wxSOCKET_INPUT_FLAG;
381 GetSock()->SetNotify(notify_flags);
382 GetSock()->Notify(TRUE);
383 GetSock()->SetTimeout(1);
385 SetBrxConnectEvent(
false);
386 GetSocketTimer()->Start(100, wxTIMER_ONE_SHOT);
390 SetConnectTime(wxDateTime::Now());
393void CommDriverN2KNet::OnSocketReadWatchdogTimer(wxTimerEvent& event) {
396 if (m_dog_value <= 0) {
397 if (GetParams().NoDataReconnect) {
399 if (GetProtocol() == TCP) {
400 wxSocketClient* tcp_socket =
dynamic_cast<wxSocketClient*
>(GetSock());
401 if (tcp_socket) tcp_socket->Close();
403 int n_reconnect_delay = wxMax(N_DOG_TIMEOUT - 2, 2);
404 wxLogMessage(wxString::Format(
" Reconnection scheduled in %d seconds.",
406 GetSocketTimer()->Start(n_reconnect_delay * 1000, wxTIMER_ONE_SHOT);
409 GetSocketThreadWatchdogTimer()->Stop();
415void CommDriverN2KNet::OnTimerSocket() {
417 wxSocketClient* tcp_socket =
dynamic_cast<wxSocketClient*
>(GetSock());
419 if (tcp_socket->IsDisconnected()) {
420 wxLogDebug(
" Attempting reconnection...");
421 SetBrxConnectEvent(
false);
423 GetSocketThreadWatchdogTimer()->Stop();
424 tcp_socket->Connect(GetAddr(), FALSE);
427 int n_reconnect_delay = N_DOG_TIMEOUT;
428 GetSocketTimer()->Start(n_reconnect_delay * 1000, wxTIMER_ONE_SHOT);
433void CommDriverN2KNet::HandleResume() {
435 wxSocketClient* tcp_socket =
dynamic_cast<wxSocketClient*
>(GetSock());
437 GetSocketThreadWatchdogTimer()->Stop();
442 int n_reconnect_delay = wxMax(N_DOG_TIMEOUT - 2, 2);
443 wxLogMessage(wxString::Format(
" Reconnection scheduled in %d seconds.",
446 GetSocketTimer()->Start(n_reconnect_delay * 1000, wxTIMER_ONE_SHOT);
450bool CommDriverN2KNet::SendMessage(std::shared_ptr<const NavMsg> msg,
451 std::shared_ptr<const NavAddr> addr) {
452 if (!msg)
return false;
453 auto msg_n2k = std::dynamic_pointer_cast<const Nmea2000Msg>(msg);
454 auto dest_addr_n2k = std::static_pointer_cast<const NavAddr2000>(addr);
455 return SendN2KNetwork(msg_n2k, dest_addr_n2k);
458std::vector<unsigned char> CommDriverN2KNet::PrepareLogPayload(
459 std::shared_ptr<const Nmea2000Msg>& msg,
460 std::shared_ptr<const NavAddr2000> addr) {
461 std::vector<unsigned char> data;
462 data.push_back(0x94);
463 data.push_back(0x13);
464 data.push_back(msg->priority);
465 data.push_back(msg->PGN.pgn & 0xFF);
466 data.push_back((msg->PGN.pgn >> 8) & 0xFF);
467 data.push_back((msg->PGN.pgn >> 16) & 0xFF);
468 data.push_back(addr->address);
469 data.push_back(addr->address);
470 for (
size_t n = 0; n < msg->payload.size(); n++)
471 data.push_back(msg->payload[n]);
472 data.push_back(0x55);
476std::vector<unsigned char> CommDriverN2KNet::PushCompleteMsg(
477 const CanHeader header,
int position,
const can_frame frame) {
478 std::vector<unsigned char> data;
479 data.push_back(0x93);
480 data.push_back(0x13);
481 data.push_back(header.priority);
482 data.push_back(header.pgn & 0xFF);
483 data.push_back((header.pgn >> 8) & 0xFF);
484 data.push_back((header.pgn >> 16) & 0xFF);
485 data.push_back(header.destination);
486 data.push_back(header.source);
487 data.push_back(0xFF);
488 data.push_back(0xFF);
489 data.push_back(0xFF);
490 data.push_back(0xFF);
491 data.push_back(CAN_MAX_DLEN);
492 for (
size_t n = 0; n < CAN_MAX_DLEN; n++) data.push_back(frame.data[n]);
493 data.push_back(0x55);
497std::vector<unsigned char> CommDriverN2KNet::PushFastMsgFragment(
499 std::vector<unsigned char> data;
500 data.push_back(0x93);
501 data.push_back(fast_messages->entries[position].expected_length + 11);
502 data.push_back(header.priority);
503 data.push_back(header.pgn & 0xFF);
504 data.push_back((header.pgn >> 8) & 0xFF);
505 data.push_back((header.pgn >> 16) & 0xFF);
506 data.push_back(header.destination);
507 data.push_back(header.source);
508 data.push_back(0xFF);
509 data.push_back(0xFF);
510 data.push_back(0xFF);
511 data.push_back(0xFF);
512 data.push_back(fast_messages->entries[position].expected_length);
513 for (
size_t n = 0; n < fast_messages->entries[position].expected_length; n++)
514 data.push_back(fast_messages->entries[position].data[n]);
515 data.push_back(0x55);
516 fast_messages->
Remove(position);
526void CommDriverN2KNet::HandleCanFrameInput(can_frame frame) {
533 if (position == kNotFound) {
540 if ((frame.data[0] & 0x1F) == 0) {
542 ready = fast_messages->
InsertEntry(header, frame.data, position);
547 ready = fast_messages->
AppendEntry(header, frame.data, position);
551 std::vector<unsigned char> vec;
554 vec = PushFastMsgFragment(header, position);
557 vec = PushCompleteMsg(header, position, frame);
561 if (HandleMgntMsg(header.pgn, vec))
return;
565 auto payload = std::make_shared<std::vector<uint8_t>>(vec);
566 Nevent.SetPayload(payload);
567 AddPendingEvent(Nevent);
571static bool isASCII(
const std::vector<unsigned char>& packet) {
572 for (
unsigned char c : packet) {
573 if (!isascii(c))
return false;
578N2K_Format CommDriverN2KNet::DetectFormat(
579 const std::vector<unsigned char>& packet) {
584 if (isASCII(packet)) {
585 std::string payload = std::string(packet.begin(), packet.end());
586 if (payload.find(
"$PCDIN") != std::string::npos) {
587 return N2KFormat_SeaSmart;
588 }
else if (payload.find(
"$MXPGN") != std::string::npos) {
592 return N2KFormat_MiniPlex;
593 }
else if (std::find(packet.begin(), packet.end(),
':') != packet.end()) {
594 return N2KFormat_Actisense_RAW_ASCII;
596 return N2KFormat_Actisense_N2K_ASCII;
599 if (packet[2] == 0x95)
600 return N2KFormat_Actisense_RAW;
601 else if (packet[2] == 0xd0)
602 return N2KFormat_Actisense_N2K;
603 else if (packet[2] == 0x93)
604 return N2KFormat_Actisense_NGT;
606 return N2KFormat_Undefined;
609bool CommDriverN2KNet::ProcessActisense_N2K(std::vector<unsigned char> packet) {
612 std::vector<unsigned char> data;
614 bool bGotESC =
false;
615 bool bGotSOT =
false;
618 uint8_t next_byte = m_circle.
Get();
622 if (next_byte == ESCAPE) {
623 data.push_back(next_byte);
625 }
else if (next_byte == ENDOFTEXT) {
629 unsigned int msg_length =
630 (uint32_t)data[1] + ((uint32_t)data[2] << 8);
633 if (msg_length == data.size() - 1) {
634 uint8_t destination = data[3];
635 uint8_t source = data[4];
637 uint8_t dprp = data[7];
640 uint8_t rAndDP = dprp & 3;
643 uint8_t pduFormat = data[6];
644 uint32_t pgn = (rAndDP << 16) + (pduFormat << 8);
650 std::vector<uint8_t> o_payload;
651 o_payload.push_back(0x93);
652 o_payload.push_back(0x13);
653 o_payload.push_back(priority);
654 o_payload.push_back(pgn & 0xFF);
655 o_payload.push_back((pgn >> 8) & 0xFF);
656 o_payload.push_back((pgn >> 16) & 0xFF);
657 o_payload.push_back(destination);
658 o_payload.push_back(source);
659 o_payload.push_back(0xFF);
660 o_payload.push_back(0xFF);
661 o_payload.push_back(0xFF);
662 o_payload.push_back(0xFF);
663 o_payload.push_back(data.size());
666 for (
size_t n = 13; n < data.size() - 1; n++)
667 o_payload.push_back(data[n]);
669 o_payload.push_back(0x55);
674 std::make_shared<std::vector<uint8_t>>(o_payload);
675 Nevent.SetPayload(n2k_payload);
676 AddPendingEvent(Nevent);
683 }
else if (next_byte == STARTOFTEXT) {
692 bGotESC = (next_byte == ESCAPE);
695 data.push_back(next_byte);
701 if (STARTOFTEXT == next_byte) {
707 bGotESC = (next_byte == ESCAPE);
712 data.push_back(next_byte);
721bool CommDriverN2KNet::ProcessActisense_RAW(std::vector<unsigned char> packet) {
726 std::vector<unsigned char> data;
728 bool bGotESC =
false;
729 bool bGotSOT =
false;
732 uint8_t next_byte = m_circle.
Get();
736 if (next_byte == ESCAPE) {
737 data.push_back(next_byte);
739 }
else if (next_byte == ENDOFTEXT) {
744 if (data.size() >= 8) {
745 size_t dLen = data[1];
747 if (dLen + 3 == data.size()) {
749 memcpy(&frame.can_id, &data.data()[4], 4);
752 memcpy(&frame.data, &data.data()[8], 8);
754 HandleCanFrameInput(frame);
762 }
else if (next_byte == STARTOFTEXT) {
771 bGotESC = (next_byte == ESCAPE);
774 data.push_back(next_byte);
780 if (STARTOFTEXT == next_byte) {
786 bGotESC = (next_byte == ESCAPE);
791 data.push_back(next_byte);
800bool CommDriverN2KNet::ProcessActisense_NGT(std::vector<unsigned char> packet) {
801 std::vector<unsigned char> data;
803 bool bGotESC =
false;
804 bool bGotSOT =
false;
807 uint8_t next_byte = m_circle.
Get();
811 if (next_byte == ESCAPE) {
812 data.push_back(next_byte);
814 }
else if (next_byte == ENDOFTEXT) {
817 auto n2k_payload = std::make_shared<std::vector<uint8_t>>(data);
818 Nevent.SetPayload(n2k_payload);
819 AddPendingEvent(Nevent);
825 }
else if (next_byte == STARTOFTEXT) {
834 bGotESC = (next_byte == ESCAPE);
837 data.push_back(next_byte);
843 if (STARTOFTEXT == next_byte) {
849 bGotESC = (next_byte == ESCAPE);
854 data.push_back(next_byte);
863bool CommDriverN2KNet::ProcessActisense_ASCII_RAW(
864 std::vector<unsigned char> packet) {
868 char b = m_circle.
Get();
869 if ((b != 0x0a) && (b != 0x0d)) {
877 wxString ss(m_sentence.c_str());
879 wxStringTokenizer tkz(ss,
" ");
882 wxString token = tkz.GetNextToken();
884 token = tkz.GetNextToken();
886 m_TX_flag = token[0];
889 token = tkz.GetNextToken();
891 token.ToLong(&canID, 16);
892 frame.can_id = canID;
895 unsigned char bytes[8];
897 for (
unsigned int i = 0; i < 8; i++) {
898 if (tkz.HasMoreTokens()) {
899 token = tkz.GetNextToken();
901 token.ToLong(&tui, 16);
902 bytes[i] = (uint8_t)tui;
905 memcpy(&frame.data, bytes, 8);
906 HandleCanFrameInput(frame);
912bool CommDriverN2KNet::ProcessActisense_ASCII_N2K(
913 std::vector<unsigned char> packet) {
915 std::string sentence;
918 char b = m_circle.
Get();
919 if ((b != 0x0a) && (b != 0x0d)) {
927 wxString ss(sentence.c_str());
928 wxStringTokenizer tkz(ss,
" ");
932 wxString time_header = tkz.GetNextToken();
934 wxString sprio_addr = tkz.GetNextToken();
936 sprio_addr.ToLong(&prio_addr, 16);
937 uint8_t priority = (uint8_t)prio_addr & 0X0F;
938 uint8_t destination = (uint8_t)(prio_addr >> 4) & 0X0FF;
939 uint8_t source = (uint8_t)(prio_addr >> 12) & 0X0FF;
942 wxString sPGN = tkz.GetNextToken();
944 sPGN.ToULong(&PGN, 16);
948 wxString sdata = tkz.GetNextToken();
949 std::vector<uint8_t> data;
950 for (
size_t i = 0; i < sdata.Length(); i += 2) {
952 wxString stui = sdata.Mid(i, 2);
953 stui.ToLong(&dv, 16);
954 data.push_back((uint8_t)dv);
958 std::vector<uint8_t> o_payload;
959 o_payload.push_back(0x93);
960 o_payload.push_back(0x13);
961 o_payload.push_back(priority);
962 o_payload.push_back(PGN & 0xFF);
963 o_payload.push_back((PGN >> 8) & 0xFF);
964 o_payload.push_back((PGN >> 16) & 0xFF);
965 o_payload.push_back(destination);
966 o_payload.push_back(source);
967 o_payload.push_back(0xFF);
968 o_payload.push_back(0xFF);
969 o_payload.push_back(0xFF);
970 o_payload.push_back(0xFF);
971 o_payload.push_back(data.size());
972 for (
size_t n = 0; n < data.size(); n++) o_payload.push_back(data[n]);
973 o_payload.push_back(0x55);
975 if (HandleMgntMsg(PGN, o_payload))
return false;
979 auto n2k_payload = std::make_shared<std::vector<uint8_t>>(o_payload);
980 Nevent.SetPayload(n2k_payload);
981 AddPendingEvent(Nevent);
987bool CommDriverN2KNet::ProcessSeaSmart(std::vector<unsigned char> packet) {
989 char b = m_circle.
Get();
990 if ((b != 0x0a) && (b != 0x0d)) {
998 wxString ss(m_sentence.c_str());
1000 wxStringTokenizer tkz(ss,
",");
1003 wxString token = tkz.GetNextToken();
1006 token = tkz.GetNextToken();
1008 token.ToULong(&PGN, 16);
1010 token = tkz.GetNextToken();
1011 unsigned long timestamp;
1012 token.ToULong(×tamp, 16);
1014 token = tkz.GetNextToken();
1015 unsigned long source;
1016 token.ToULong(&source, 16);
1018 token = tkz.GetNextToken();
1020 wxStringTokenizer datatkz(token,
"*");
1021 wxString data = datatkz.GetNextToken();
1024 std::vector<uint8_t> o_payload;
1025 o_payload.push_back(0x93);
1026 o_payload.push_back(0x13);
1027 o_payload.push_back(3);
1028 o_payload.push_back(PGN & 0xFF);
1029 o_payload.push_back((PGN >> 8) & 0xFF);
1030 o_payload.push_back((PGN >> 16) & 0xFF);
1031 o_payload.push_back(0xFF);
1032 o_payload.push_back((uint8_t)source);
1033 o_payload.push_back(timestamp & 0xFF);
1034 o_payload.push_back((timestamp >> 8) & 0xFF);
1035 o_payload.push_back((timestamp >> 16) & 0xFF);
1036 o_payload.push_back((timestamp >> 24) & 0xFF);
1037 o_payload.push_back((uint8_t)data.Length() / 2);
1038 for (
size_t i = 0; i < data.Length(); i += 2) {
1040 wxString sbyte = data.Mid(i, 2);
1041 sbyte.ToULong(&dv, 16);
1042 o_payload.push_back((uint8_t)dv);
1044 o_payload.push_back(0x55);
1046 if (HandleMgntMsg(PGN, o_payload))
return false;
1050 auto n2k_payload = std::make_shared<std::vector<uint8_t>>(o_payload);
1051 Nevent.SetPayload(n2k_payload);
1052 AddPendingEvent(Nevent);
1058bool CommDriverN2KNet::ProcessMiniPlex(std::vector<unsigned char> packet) {
1154 char b = m_circle.
Get();
1155 if ((b != 0x0a) && (b != 0x0d)) {
1163 wxString ss(m_sentence.c_str());
1165 wxStringTokenizer tkz(ss,
",");
1168 wxString token = tkz.GetNextToken();
1171 token = tkz.GetNextToken();
1173 token.ToULong(&PGN, 16);
1175 token = tkz.GetNextToken();
1177 token.ToULong(&attr, 16);
1179 bool send_bit = (attr >> 15) != 0;
1181 uint8_t priority = (attr >> 12) & 0x07;
1184 uint8_t dlc = (attr >> 8) & 0x0F;
1187 uint8_t address = attr & 0xFF;
1189 token = tkz.GetNextToken();
1191 wxStringTokenizer datatkz(token,
"*");
1192 wxString data = datatkz.GetNextToken();
1200 memset(&frame.data, 0, 8);
1201 for (
size_t i = 0; i < data.Length(); i += 2) {
1203 wxString sbyte = data.Mid(data.Length() - i - 2, 2);
1204 sbyte.ToULong(&dv, 16);
1205 frame.data[i / 2] = ((uint8_t)dv);
1207 frame.can_id = (uint32_t)BuildCanID(priority, address, 0xFF, PGN);
1208 HandleCanFrameInput(frame);
1215#define RD_BUF_SIZE 4096
1218 switch (event.GetSocketEvent()) {
1219 case wxSOCKET_INPUT: {
1233 std::vector<unsigned char> data(RD_BUF_SIZE + 1);
1235 uint8_t next_byte = 0;
1237 event.GetSocket()->Read(&data.front(), RD_BUF_SIZE);
1238 if (!event.GetSocket()->Error()) {
1239 m_driver_stats.available =
true;
1240 size_t count =
event.GetSocket()->LastCount();
1253 for (
int i = 0; i < newdata; i++) {
1254 if (!m_circle.
IsFull()) m_circle.
Put(data[i]);
1259 m_n2k_format = DetectFormat(data);
1261 switch (m_n2k_format) {
1262 case N2KFormat_Actisense_RAW_ASCII:
1263 ProcessActisense_ASCII_RAW(data);
1265 case N2KFormat_YD_RAW:
1266 ProcessActisense_ASCII_RAW(data);
1268 case N2KFormat_Actisense_N2K_ASCII:
1269 ProcessActisense_ASCII_N2K(data);
1271 case N2KFormat_Actisense_N2K:
1272 ProcessActisense_N2K(data);
1274 case N2KFormat_Actisense_RAW:
1275 ProcessActisense_RAW(data);
1277 case N2KFormat_Actisense_NGT:
1278 ProcessActisense_NGT(data);
1280 case N2KFormat_SeaSmart:
1281 ProcessSeaSmart(data);
1283 case N2KFormat_MiniPlex:
1284 ProcessMiniPlex(data);
1286 case N2KFormat_Undefined:
1293 m_dog_value = N_DOG_TIMEOUT;
1297 case wxSOCKET_LOST: {
1298 m_driver_stats.available =
false;
1299 if (GetProtocol() == TCP || GetProtocol() == GPSD) {
1300 if (GetBrxConnectEvent())
1301 wxLogMessage(wxString::Format(
"NetworkDataStream connection lost: %s",
1302 GetPort().c_str()));
1303 if (GetSockServer()) {
1304 GetSock()->Destroy();
1308 wxDateTime now = wxDateTime::Now();
1309 wxTimeSpan since_connect(
1311 if (GetConnectTime().IsValid()) since_connect = now - GetConnectTime();
1313 int retry_time = 5000;
1319 if (!GetBrxConnectEvent() && (since_connect.GetSeconds() < 5))
1322 GetSocketThreadWatchdogTimer()->Stop();
1323 GetSocketTimer()->Start(
1324 retry_time, wxTIMER_ONE_SHOT);
1329 case wxSOCKET_CONNECTION: {
1330 m_driver_stats.available =
true;
1331 if (GetProtocol() == GPSD) {
1335 char cmd[] =
"?WATCH={\"class\":\"WATCH\", \"nmea\":true}";
1336 GetSock()->Write(cmd, strlen(cmd));
1337 }
else if (GetProtocol() == TCP) {
1339 wxString::Format(
"TCP NetworkDataStream connection established: %s",
1340 GetPort().c_str()));
1341 m_dog_value = N_DOG_TIMEOUT;
1342 if (GetPortType() != DS_TYPE_OUTPUT) {
1344 if (GetParams().NoDataReconnect)
1345 GetSocketThreadWatchdogTimer()->Start(1000);
1347 if (GetPortType() != DS_TYPE_INPUT && GetSock()->IsOk())
1348 (void)SetOutputSocketOptions(GetSock());
1349 GetSocketTimer()->Stop();
1350 SetBrxConnectEvent(
true);
1353 SetConnectTime(wxDateTime::Now());
1362void CommDriverN2KNet::OnServerSocketEvent(wxSocketEvent& event) {
1363 switch (event.GetSocketEvent()) {
1364 case wxSOCKET_CONNECTION: {
1365 m_driver_stats.available =
true;
1366 SetSock(GetSockServer()->Accept(
false));
1369 GetSock()->SetTimeout(2);
1371 GetSock()->SetEventHandler(*
this, DS_SOCKET_ID);
1372 int notify_flags = (wxSOCKET_CONNECTION_FLAG | wxSOCKET_LOST_FLAG);
1373 if (GetPortType() != DS_TYPE_INPUT) {
1374 notify_flags |= wxSOCKET_OUTPUT_FLAG;
1375 (void)SetOutputSocketOptions(GetSock());
1377 if (GetPortType() != DS_TYPE_OUTPUT)
1378 notify_flags |= wxSOCKET_INPUT_FLAG;
1379 GetSock()->SetNotify(notify_flags);
1380 GetSock()->Notify(
true);
1391std::vector<unsigned char> MakeSimpleOutMsg(
1392 int data_format,
int pgn, std::vector<unsigned char>& payload) {
1393 std::vector<unsigned char> out_vec;
1395 switch (data_format) {
1396 case N2KFormat_YD_RAW:
1397 case N2KFormat_Actisense_RAW_ASCII: {
1399 unsigned can_id = BuildCanID(6, 0xff, 0xff, pgn);
1400 std::stringstream ss;
1401 ss << std::setfill(
'0') << std::setw(8) << std::hex << can_id;
1402 for (
unsigned char s : ss.str()) out_vec.push_back(s);
1403 out_vec.push_back(
' ');
1408 for (
unsigned char d : payload) {
1409 snprintf(tv, 4,
"%02X ", d);
1412 for (
unsigned char s : sspl) out_vec.push_back(s);
1416 out_vec.push_back(0x0d);
1417 out_vec.push_back(0x0a);
1420 case N2KFormat_Actisense_N2K_ASCII: {
1422 wxDateTime now = wxDateTime::Now();
1423 wxString stime = now.Format(
"%H%M%S");
1425 std::string sstime = stime.ToStdString();
1426 out_vec.push_back(
'A');
1427 for (
unsigned char s : sstime) out_vec.push_back(s);
1431 sdp.Printf(
"%02X%02X%1X ",
1433 (
unsigned char)0xFF, 0x6);
1434 std::string ssdp = sdp.ToStdString();
1435 for (
unsigned char s : ssdp) out_vec.push_back(s);
1439 spgn.Printf(
"%05X ", pgn);
1440 std::string sspgn = spgn.ToStdString();
1441 for (
unsigned char s : sspgn) out_vec.push_back(s);
1446 for (
unsigned char d : payload) {
1447 snprintf(tv, 3,
"%02X", d);
1450 for (
unsigned char s : sspl) out_vec.push_back(s);
1453 out_vec.push_back(0x0d);
1454 out_vec.push_back(0x0a);
1457 case N2KFormat_MiniPlex: {
1458 out_vec.push_back(
'$');
1459 out_vec.push_back(
'M');
1460 out_vec.push_back(
'X');
1461 out_vec.push_back(
'P');
1462 out_vec.push_back(
'G');
1463 out_vec.push_back(
'N');
1464 out_vec.push_back(
',');
1467 spgn.Printf(
"%06X,", pgn);
1468 std::string sspgn = spgn.ToStdString();
1469 for (
unsigned char c : sspgn) {
1470 out_vec.push_back(c);
1476 attr |= ((uint16_t)0x06) << 12;
1477 attr |= ((uint16_t)payload.size()) << 8;
1478 attr |= (uint16_t)0xFF;
1482 sattr.Printf(
"%04X,", attr);
1483 std::string ssattr = sattr.ToStdString();
1484 for (
unsigned char c : ssattr) {
1485 out_vec.push_back(c);
1489 for (
auto rit = payload.rbegin(); rit != payload.rend(); ++rit) {
1490 snprintf(tv, 3,
"%02X", *rit);
1491 out_vec.push_back(tv[0]);
1492 out_vec.push_back(tv[1]);
1496 for (
auto ci = ++out_vec.begin(); ci != out_vec.end(); ci++) {
1499 out_vec.push_back(
'*');
1500 snprintf(tv, 3,
"%02X", crc);
1501 out_vec.push_back(tv[0]);
1502 out_vec.push_back(tv[1]);
1505 out_vec.push_back(0x0d);
1506 out_vec.push_back(0x0a);
1517std::vector<std::vector<unsigned char>> CommDriverN2KNet::GetTxVector(
1518 const std::shared_ptr<const Nmea2000Msg>& msg,
1519 std::shared_ptr<const NavAddr2000> dest_addr) {
1520 std::vector<std::vector<unsigned char>> tx_vector;
1523 switch (m_n2k_format) {
1524 case N2KFormat_YD_RAW:
1526 case N2KFormat_Actisense_RAW_ASCII: {
1528 if (!IsFastMessagePGN(msg->PGN.pgn) && msg->payload.size() <= 8) {
1530 std::vector<unsigned char> header_vec;
1531 std::vector<unsigned char> out_vec;
1536 unsigned long can_id =
1537 BuildCanID(msg->priority, 0, dest_addr->address, msg->PGN.pgn);
1539 std::stringstream ss;
1540 ss << std::setfill(
'0') << std::setw(8) << std::hex << can_id;
1541 for (
unsigned char s : ss.str()) header_vec.push_back(s);
1542 header_vec.push_back(
' ');
1545 for (
unsigned char s : header_vec) out_vec.push_back(s);
1549 for (
unsigned int k = 0; k < msg->payload.size(); k++) {
1551 snprintf(tb, 4,
"%02X ", msg->payload.data()[k]);
1554 for (
unsigned char s : ssdata) out_vec.push_back(s);
1557 out_vec.push_back(0x0d);
1558 out_vec.push_back(0x0a);
1560 tx_vector.push_back(out_vec);
1562 std::vector<unsigned char> header_vec;
1563 std::vector<unsigned char> out_vec;
1568 wxDateTime now = wxDateTime::Now();
1569 wxString stime = now.Format(
"%H:%M:%S");
1571 std::string sstime = stime.ToStdString();
1572 for (
unsigned char s : sstime) header_vec.push_back(s);
1575 header_vec.push_back(
'T');
1576 header_vec.push_back(
' ');
1582 unsigned long can_id =
1583 BuildCanID(msg->priority, 0, dest_addr->address, msg->PGN.pgn);
1584 std::stringstream ss;
1585 ss << std::setfill(
'0') << std::setw(8) << std::hex << can_id;
1586 for (
unsigned char s : ss.str()) header_vec.push_back(s);
1587 header_vec.push_back(
' ');
1590 int payload_size = msg->payload.size();
1591 unsigned char temp[8];
1594 (payload_size > 6 ? (payload_size - 6 - 1) / 7 + 1 + 1 : 1);
1596 for (
int i = 0; i < nframes && result; i++) {
1597 temp[0] = i | m_order;
1599 temp[1] = msg->payload.size();
1601 for (
int j = 2; j < 8; j++) {
1602 temp[j] = msg->payload.data()[cur];
1608 for (; j < 8 && cur < payload_size; j++) {
1609 temp[j] = msg->payload.data()[cur];
1612 for (; j < 8; j++) {
1620 for (
unsigned char s : header_vec) out_vec.push_back(s);
1624 for (
unsigned int k = 0; k < 8; k++) {
1626 snprintf(tb, 4,
"%02X ", temp[k]);
1629 for (
unsigned char s : ssdata) out_vec.push_back(s);
1632 out_vec.push_back(0x0d);
1633 out_vec.push_back(0x0a);
1635 tx_vector.push_back(out_vec);
1639 case N2KFormat_Actisense_N2K_ASCII: {
1656 std::vector<unsigned char> ovec;
1659 wxDateTime now = wxDateTime::Now();
1660 wxString stime = now.Format(
"%H%M%S");
1662 std::string sstime = stime.ToStdString();
1663 ovec.push_back(
'A');
1664 for (
unsigned char s : sstime) ovec.push_back(s);
1668 sdp.Printf(
"%02X%02X%1X ",
1670 (
unsigned char)dest_addr->address,
1671 (
unsigned char)msg->priority);
1672 std::string ssdp = sdp.ToStdString();
1673 for (
unsigned char s : ssdp) ovec.push_back(s);
1677 spgn.Printf(
"%05X ", (
int)msg->PGN.pgn);
1678 std::string sspgn = spgn.ToStdString();
1679 for (
unsigned char s : sspgn) ovec.push_back(s);
1684 for (
unsigned char d : msg->payload) {
1685 snprintf(tv, 3,
"%02X", d);
1688 for (
unsigned char s : sspl) ovec.push_back(s);
1691 ovec.push_back(0x0d);
1692 ovec.push_back(0x0a);
1695 tx_vector.push_back(ovec);
1699 case N2KFormat_MiniPlex: {
1700 std::vector<unsigned char> ovec;
1701 if (!IsFastMessagePGN(msg->PGN.pgn) && msg->payload.size() < 8) {
1706 (msg->payload.size() > 6 ? (msg->payload.size() - 6 - 1) / 7 + 1 + 1
1708 for (
size_t i = 0; i < nframes; i++) {
1709 ovec.push_back(
'$');
1710 ovec.push_back(
'M');
1711 ovec.push_back(
'X');
1712 ovec.push_back(
'P');
1713 ovec.push_back(
'G');
1714 ovec.push_back(
'N');
1715 ovec.push_back(
',');
1718 spgn.Printf(
"%06X,", (
int)msg->PGN.pgn);
1719 std::string sspgn = spgn.ToStdString();
1720 for (
unsigned char c : sspgn) {
1726 if (i == nframes - 1) {
1731 attr |= ((uint16_t)((uint8_t)msg->priority & 0x07)) << 12;
1732 attr |= ((uint16_t)len) << 8;
1733 attr |= (uint16_t)dest_addr->address;
1737 sattr.Printf(
"%04X,", attr);
1738 std::string ssattr = sattr.ToStdString();
1739 for (
unsigned char c : ssattr) {
1744 uint8_t databytes = i == 0 ? len - 2 : len - 1;
1745 std::vector<unsigned char> payload;
1746 for (uint8_t j = 0; j < databytes; j++) {
1747 payload.push_back(msg->payload[cur]);
1752 int psize = payload.size();
1753 while ((i > 0) && (psize < 7)) {
1754 ovec.push_back(
'F');
1755 ovec.push_back(
'F');
1760 for (
auto rit = payload.rbegin(); rit != payload.rend(); ++rit) {
1761 snprintf(tv, 3,
"%02X", *rit);
1762 ovec.push_back(tv[0]);
1763 ovec.push_back(tv[1]);
1766 snprintf(tv, 3,
"%02X", (uint8_t)msg->payload.size());
1767 ovec.push_back(tv[0]);
1768 ovec.push_back(tv[1]);
1771 snprintf(tv, 3,
"%02X", (uint8_t)i | m_order);
1772 ovec.push_back(tv[0]);
1773 ovec.push_back(tv[1]);
1777 for (
auto ci = ++ovec.begin(); ci != ovec.end(); ci++) {
1780 ovec.push_back(
'*');
1781 snprintf(tv, 3,
"%02X", crc);
1782 ovec.push_back(tv[0]);
1783 ovec.push_back(tv[1]);
1786 ovec.push_back(0x0d);
1787 ovec.push_back(0x0a);
1793 tx_vector.push_back(ovec);
1799 case N2KFormat_Actisense_N2K:
1801 case N2KFormat_Actisense_RAW:
1803 case N2KFormat_Actisense_NGT:
1805 case N2KFormat_SeaSmart:
1812 m_order = (m_order + 0x20) & 0xE0;
1817bool CommDriverN2KNet::PrepareForTX() {
1827 bool b_found =
false;
1834 if (m_n2k_format == N2KFormat_Actisense_N2K_ASCII)
return true;
1841 if (m_n2k_format == N2KFormat_MiniPlex)
return true;
1846 if (m_n2k_format == N2KFormat_SeaSmart)
return false;
1859 std::vector<unsigned char> payload;
1860 payload.push_back(0x14);
1861 payload.push_back(0xF0);
1862 payload.push_back(0x01);
1864 std::vector<std::vector<unsigned char>> out_data;
1865 std::vector<unsigned char> msg_vec =
1866 MakeSimpleOutMsg(N2KFormat_YD_RAW, 59904, payload);
1867 out_data.push_back(msg_vec);
1868 SendSentenceNetwork(out_data);
1871 m_prodinfo_timer.Start(200,
true);
1878bool CommDriverN2KNet::SendN2KNetwork(std::shared_ptr<const Nmea2000Msg>& msg,
1879 std::shared_ptr<const NavAddr2000> addr) {
1882 std::vector<std::vector<unsigned char>> out_data = GetTxVector(msg, addr);
1883 SendSentenceNetwork(out_data);
1884 m_driver_stats.
tx_count += msg->payload.size();
1887 std::vector<unsigned char> msg_payload = PrepareLogPayload(msg, addr);
1889 std::make_shared<const Nmea2000Msg>(msg->PGN.pgn, msg_payload, addr));
1894bool CommDriverN2KNet::SendSentenceNetwork(
1895 std::vector<std::vector<unsigned char>> payload) {
1902 wxDatagramSocket* udp_socket;
1903 switch (GetProtocol()) {
1905 for (std::vector<unsigned char>& v : payload) {
1906 if (GetSock() && GetSock()->IsOk()) {
1907 m_driver_stats.available =
true;
1909 GetSock()->Write(v.data(), v.size());
1910 m_dog_value = N_DOG_TIMEOUT;
1911 if (GetSock()->Error()) {
1912 if (GetSockServer()) {
1913 GetSock()->Destroy();
1916 wxSocketClient* tcp_socket =
1917 dynamic_cast<wxSocketClient*
>(GetSock());
1918 if (tcp_socket) tcp_socket->Close();
1919 if (!GetSocketTimer()->IsRunning())
1920 GetSocketTimer()->Start(
1921 5000, wxTIMER_ONE_SHOT);
1922 GetSocketThreadWatchdogTimer()->Stop();
1928 m_driver_stats.available =
false;
1935 udp_socket =
dynamic_cast<wxDatagramSocket*
>(GetTSock());
1936 if (udp_socket && udp_socket->IsOk()) {
1937 udp_socket->SendTo(GetAddr(), payload.mb_str(), payload.size());
1938 if (udp_socket->Error()) ret =
false;
1953void CommDriverN2KNet::Close() {
1954 wxLogMessage(wxString::Format(
"Closing NMEA NetworkDataStream %s",
1955 GetNetPort().c_str()));
1956 m_stats_timer.Stop();
1960 m_sock->SetOption(IPPROTO_IP, IP_DROP_MEMBERSHIP, &m_mrq_container->m_mrq,
1961 sizeof(m_mrq_container->m_mrq));
1962 m_sock->Notify(FALSE);
1964 m_driver_stats.available =
false;
1968 m_tsock->Notify(FALSE);
1972 if (m_socket_server) {
1973 m_socket_server->Notify(FALSE);
1974 m_socket_server->Destroy();
1977 m_socket_timer.Stop();
1978 m_socketread_watchdog_timer.Stop();
1981bool CommDriverN2KNet::SetOutputSocketOptions(wxSocketBase* tsock) {
1990 int nagleDisable = 1;
1991 ret = tsock->SetOption(IPPROTO_TCP, TCP_NODELAY, &nagleDisable,
1992 sizeof(nagleDisable));
1998 unsigned long outbuf_size = 1024;
1999 return (tsock->SetOption(SOL_SOCKET, SO_SNDBUF, &outbuf_size,
2000 sizeof(outbuf_size)) &&
bool IsEmpty() const noexcept
Return true if buffer is empty.
T Get()
Get item from buff; throw BufferError if empty.
void Put(const T &item)
Add item to buffer; throw BufferError if full.
bool IsFull() const noexcept
Return true if buffer is full.
void OnSocketEvent(wxSocketEvent &event)
Interface for handling incoming messages.
virtual void Notify(std::shared_ptr< const NavMsg > message)=0
Handle a received message.
Track fast message fragments eventually forming complete messages.
int AddNewEntry(void)
Allocate a new, fresh entry and return index to it.
void Remove(int pos)
Remove entry at pos.
bool AppendEntry(const CanHeader hdr, const unsigned char *data, int index)
Append fragment to existing multipart message.
int FindMatchingEntry(const CanHeader header, const unsigned char sid)
Setter.
bool InsertEntry(const CanHeader header, const unsigned char *data, int index)
Insert a new entry, first part of a multipart message.
Custom event class for OpenCPN's notification system.
Nmea2000 IP network driver.
Driver registration container, a singleton.
Raw messages layer, supports sending and recieving navmsg messages.
std::string DsPortTypeToString(dsPortType type)
Return textual representation for use in driver ioDirection attribute.
GUI constant definitions.
unsigned tx_count
Number of bytes sent since program start.
unsigned rx_count
Number of bytes received since program start.
Suspend/resume and new devices events exchange point.