Diag-Client-Lib
doip_udp_channel_handler.cpp
Go to the documentation of this file.
1 /* Diagnostic Client library
2  * Copyright (C) 2024 Avijit Dey
3  *
4  * This Source Code Form is subject to the terms of the Mozilla Public
5  * License, v. 2.0. If a copy of the MPL was not distributed with this
6  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
7  */
8 
10 
11 #include <utility>
12 
14 #include "common/logger.h"
15 #include "core/include/span.h"
16 
17 namespace doip_client {
18 namespace channel {
19 namespace udp_channel {
20 
24 constexpr std::uint8_t kDoip_GenericHeader_IncorrectPattern{0x00};
25 constexpr std::uint8_t kDoip_GenericHeader_UnknownPayload{0x01};
26 constexpr std::uint8_t kDoip_GenericHeader_MessageTooLarge{0x02};
27 constexpr std::uint8_t kDoip_GenericHeader_OutOfMemory{0x03};
28 constexpr std::uint8_t kDoip_GenericHeader_InvalidPayloadLen{0x04};
29 
31  sockets::UdpSocketHandler &udp_socket_handler_broadcast,
32  sockets::UdpSocketHandler &udp_socket_handler_unicast, DoipUdpChannel &channel)
33  : vehicle_discovery_handler_{udp_socket_handler_broadcast, channel},
34  vehicle_identification_handler_{udp_socket_handler_unicast, channel} {}
35 
37  uds_transport::UdsMessageConstPtr vehicle_identification_request) noexcept
41  // Get the udp handler type from payload
42  std::uint8_t const handler_type{vehicle_identification_request->GetPayload()[0u]};
43  // deserialize and send to proper handler
44  switch (handler_type) {
45  case 0U:
46  // 0U -> Vehicle Identification Req
47  ret_val = vehicle_identification_handler_.HandleVehicleIdentificationRequest(
48  std::move(vehicle_identification_request));
49  break;
50  case 1U:
51  // 1U -> Power Mode Req
52  break;
53  }
54  return ret_val;
55 }
56 
57 auto DoipUdpChannelHandler::HandleMessageUnicast(UdpMessagePtr udp_rx_message) noexcept -> void {
58  std::uint8_t nack_code{};
59  DoipMessage doip_rx_message{DoipMessage::MessageType::kUdp, udp_rx_message->GetHostIpAddress(),
60  udp_rx_message->GetHostPortNumber(), udp_rx_message->GetPayload()};
61  // Process the Doip Generic header check
62  if (ProcessDoIPHeader(doip_rx_message, nack_code)) {
63  ProcessDoIPPayload(doip_rx_message);
64  } else {
65  // send NACK or ignore
66  (void) nack_code;
67  }
68 }
69 
70 auto DoipUdpChannelHandler::HandleMessageBroadcast(UdpMessagePtr udp_rx_message) noexcept -> void {
71  uint8_t nack_code;
72  DoipMessage doip_rx_message{DoipMessage::MessageType::kUdp, udp_rx_message->GetHostIpAddress(),
73  udp_rx_message->GetHostPortNumber(), udp_rx_message->GetPayload()};
74  // Process the Doip Generic header check
75  if (ProcessDoIPHeader(doip_rx_message, nack_code)) {
76  vehicle_discovery_handler_.ProcessVehicleAnnouncementResponse(doip_rx_message);
77  } else {
78  // send NACK or ignore
79  (void) nack_code;
80  }
81 }
82 
84  std::uint8_t &nack_code) noexcept -> bool {
85  bool ret_val{false};
86  /* Check the header synchronisation pattern */
87  if (((doip_rx_message.GetProtocolVersion() == kDoip_ProtocolVersion) &&
88  (doip_rx_message.GetInverseProtocolVersion() ==
89  static_cast<std::uint8_t>(~kDoip_ProtocolVersion))) ||
90  ((doip_rx_message.GetProtocolVersion() == kDoip_ProtocolVersion_Def) &&
91  (doip_rx_message.GetInverseProtocolVersion() ==
92  static_cast<std::uint8_t>(~kDoip_ProtocolVersion_Def)))) {
93  /* Check the supported payload type */
94  if (doip_rx_message.GetPayloadType() == kDoip_VehicleAnnouncement_ResType) {
95  /* Req-[AUTOSAR_SWS_DiagnosticOverIP][SWS_DoIP_00017] */
96  if (doip_rx_message.GetPayloadLength() <= kDoip_Protocol_MaxPayload) {
97  /* Req-[AUTOSAR_SWS_DiagnosticOverIP][SWS_DoIP_00018] */
98  if (doip_rx_message.GetPayloadLength() <= kUdpChannelLength) {
99  /* Req-[AUTOSAR_SWS_DiagnosticOverIP][SWS_DoIP_00019] */
100  if (ProcessDoIPPayloadLength(doip_rx_message.GetPayloadLength(),
101  doip_rx_message.GetPayloadType())) {
102  ret_val = true;
103  } else {
104  // Send NACK code 0x04, close the socket
106  // socket closure ??
107  }
108  } else {
109  // Send NACK code 0x03, discard message
111  }
112  } else {
113  // Send NACK code 0x02, discard message
115  }
116  } else { // Send NACK code 0x01, discard message
118  }
119  } else { // Send NACK code 0x00, close the socket
121  // socket closure
122  }
123  return ret_val;
124 }
125 
126 auto DoipUdpChannelHandler::ProcessDoIPPayloadLength(std::uint32_t payload_len,
127  std::uint16_t payload_type) noexcept -> bool {
128  bool ret_val{false};
129  switch (payload_type) {
131  if (payload_len <= kDoip_VehicleAnnouncement_ResMaxLen) ret_val = true;
132  break;
133  }
134  default:
135  // do nothing
136  break;
137  }
138  return ret_val;
139 }
140 
143  std::lock_guard<std::mutex> const lck(channel_handler_lock);
144  switch (doip_payload.GetPayloadType()) {
147  break;
148  }
149  default:
150  /* do nothing */
151  break;
152  }
153 }
154 
155 } // namespace udp_channel
156 } // namespace channel
157 } // namespace doip_client
Immutable class to store received doip message.
Definition: doip_message.h:21
std::uint16_t GetPayloadType() const
Get the payload type.
Definition: doip_message.h:94
RxSocketType
Definition of socket type from where the request was received.
Definition: doip_message.h:31
std::mutex channel_handler_lock
Mutex to protect critical section.
auto ProcessDoIPHeader(DoipMessage &doip_rx_message, std::uint8_t &nack_code) noexcept -> bool
Function to process doip header in received response.
sockets::UdpSocketHandler::MessagePtr UdpMessagePtr
Type alias for Tcp message pointer.
auto SendVehicleIdentificationRequest(uds_transport::UdsMessageConstPtr vehicle_identification_request) noexcept -> uds_transport::UdsTransportProtocolMgr::TransmissionResult
Function to vehicle identification request to the connected network.
void ProcessDoIPPayload(DoipMessage &doip_payload, DoipMessage::RxSocketType socket_type=DoipMessage::RxSocketType::kUnicast)
Function to process the doip payload.
VehicleIdentificationHandler vehicle_identification_handler_
Handler to process vehicle identification req/res messages.
DoipUdpChannelHandler(sockets::UdpSocketHandler &udp_socket_handler_broadcast, sockets::UdpSocketHandler &udp_socket_handler_unicast, DoipUdpChannel &channel)
Constructs an instance of DoipUdpChannelHandler.
void HandleMessageUnicast(UdpMessagePtr udp_rx_message) noexcept
Function to process the received unicast udp message.
static auto ProcessDoIPPayloadLength(std::uint32_t payload_len, std::uint16_t payload_type) noexcept -> bool
Function to verify payload length of various payload type.
void HandleMessageBroadcast(UdpMessagePtr udp_rx_message) noexcept
Function to process the received broadcast udp message.
Class to manage a udp channel as per DoIP protocol.
void ProcessVehicleIdentificationResponse(DoipMessage &doip_payload) noexcept
Function to process received vehicle identification response.
Handler class to manage different socket of various client (Udp / Tcp)
constexpr std::uint8_t kDoip_GenericHeader_IncorrectPattern
Generic DoIP Header NACK codes.
constexpr std::uint8_t kDoip_GenericHeader_OutOfMemory
constexpr std::uint8_t kDoip_GenericHeader_InvalidPayloadLen
constexpr std::uint8_t kDoip_GenericHeader_MessageTooLarge
constexpr std::uint8_t kDoip_GenericHeader_UnknownPayload
constexpr std::uint8_t kDoip_ProtocolVersion_Def
constexpr std::uint32_t kDoip_Protocol_MaxPayload
constexpr std::uint32_t kUdpChannelLength
constexpr std::uint16_t kDoip_VehicleAnnouncement_ResType
constexpr std::uint8_t kDoip_ProtocolVersion
constexpr std::uint32_t kDoip_VehicleAnnouncement_ResMaxLen
std::unique_ptr< const UdsMessage > UdsMessageConstPtr
Definition: uds_message.h:69