Diag-Client-Lib
doip_vehicle_identification_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 
13 #include "common/logger.h"
14 #include "utility/state.h"
15 #include "utility/sync_timer.h"
16 
17 namespace doip_client {
18 namespace channel {
19 namespace udp_channel {
20 namespace {
21 
25 using VehiclePayloadType = std::pair<std::uint16_t, std::uint8_t>;
26 
30 enum class VehicleIdentificationState : std::uint8_t {
31  kIdle = 0U,
35 };
36 
40 class kIdle final : public utility::state::State<VehicleIdentificationState> {
41  public:
48 
52  void Start() override {}
53 
57  void Stop() override {}
58 };
59 
64  : public utility::state::State<VehicleIdentificationState> {
65  public:
73 
77  void Start() override {}
78 
82  void Stop() override {}
83 };
84 
89  : public utility::state::State<VehicleIdentificationState> {
90  public:
98 
102  void Start() override {}
103 
107  void Stop() override {}
108 };
109 
113 class kDoIPCtrlTimeout final : public utility::state::State<VehicleIdentificationState> {
114  public:
121  : State<VehicleIdentificationState>(state) {}
122 
126  void Start() override {}
127 
131  void Stop() override {}
132 };
133 
141 auto CreateDoipGenericHeader(std::uint16_t payload_type, std::uint32_t payload_len) noexcept
142  -> std::vector<std::uint8_t> {
143  std::vector<std::uint8_t> output_buffer{};
144  output_buffer.emplace_back(kDoip_ProtocolVersion);
145  output_buffer.emplace_back(~(static_cast<std::uint8_t>(kDoip_ProtocolVersion)));
146  output_buffer.emplace_back(static_cast<std::uint8_t>((payload_type & 0xFF00) >> 8));
147  output_buffer.emplace_back(static_cast<std::uint8_t>(payload_type & 0x00FF));
148  output_buffer.emplace_back(static_cast<std::uint8_t>((payload_len & 0xFF000000) >> 24));
149  output_buffer.emplace_back(static_cast<std::uint8_t>((payload_len & 0x00FF0000) >> 16));
150  output_buffer.emplace_back(static_cast<std::uint8_t>((payload_len & 0x0000FF00) >> 8));
151  output_buffer.emplace_back(static_cast<std::uint8_t>(payload_len & 0x000000FF));
152  return output_buffer;
153 }
154 
161 auto GetVehicleIdentificationPayloadType(std::uint8_t preselection_mode) noexcept
162  -> VehiclePayloadType {
163  std::uint16_t vehicle_identification_type{0u};
164  std::uint8_t vehicle_identification_length{0u};
165  switch (preselection_mode) {
166  case 0U:
167  vehicle_identification_type = kDoip_VehicleIdentification_ReqType;
168  vehicle_identification_length = kDoip_VehicleIdentification_ReqLen;
169  break;
170  case 1U:
171  vehicle_identification_type = kDoip_VehicleIdentificationVIN_ReqType;
172  vehicle_identification_length = kDoip_VehicleIdentificationVIN_ReqLen;
173  break;
174  case 2U:
175  vehicle_identification_type = kDoip_VehicleIdentificationEID_ReqType;
176  vehicle_identification_length = kDoip_VehicleIdentificationEID_ReqLen;
177  break;
178  default:
179  break;
180  }
181  return VehiclePayloadType{vehicle_identification_type, vehicle_identification_length};
182 }
183 
184 } // namespace
185 
190  public:
196 
201 
208  DoipUdpChannel &channel)
209  : udp_socket_handler_{udp_socket_handler},
210  channel_{channel},
211  state_context_{} {
212  // create and add state for vehicle identification
213  // kIdle
214  state_context_.AddState(VehicleIdentificationState::kIdle,
215  std::make_unique<kIdle>(VehicleIdentificationState::kIdle));
216  // kSendVehicleIdentificationReq
217  state_context_.AddState(VehicleIdentificationState::kSendVehicleIdentificationReq,
218  std::make_unique<kSendVehicleIdentificationReq>(
219  VehicleIdentificationState::kSendVehicleIdentificationReq));
220  // kWaitForVehicleIdentificationRes
221  state_context_.AddState(VehicleIdentificationState::kWaitForVehicleIdentificationRes,
222  std::make_unique<kWaitForVehicleIdentificationRes>(
223  VehicleIdentificationState::kWaitForVehicleIdentificationRes));
224  // kDoIPCtrlTimeout
226  VehicleIdentificationState::kDoIPCtrlTimeout,
227  std::make_unique<kDoIPCtrlTimeout>(VehicleIdentificationState::kDoIPCtrlTimeout));
228  // Transit to idle state
229  state_context_.TransitionTo(VehicleIdentificationState::kIdle);
230  }
231 
237 
242  auto GetSocketHandler() noexcept -> sockets::UdpSocketHandler & { return udp_socket_handler_; }
243 
248  auto GetDoipChannel() noexcept -> DoipUdpChannel & { return channel_; }
249 
254  auto GetSyncTimer() noexcept -> SyncTimer & { return sync_timer_; }
255 
256  private:
261 
266 
271 
276 };
277 
279  sockets::UdpSocketHandler &udp_socket_handler, DoipUdpChannel &channel)
280  : handler_impl_{
281  std::make_unique<VehicleIdentificationHandlerImpl>(udp_socket_handler, channel)} {}
282 
284 
286  uds_transport::UdsMessageConstPtr vehicle_identification_request) noexcept
290  if (handler_impl_->GetStateContext().GetActiveState().GetState() ==
291  VehicleIdentificationState::kIdle) {
292  // change state before sending if SendVehicleIdentificationRequest call takes more time to return and in the
293  // same time async reception starts
294  handler_impl_->GetStateContext().TransitionTo(
295  VehicleIdentificationState::kWaitForVehicleIdentificationRes);
296  if (SendVehicleIdentificationRequest(std::move(vehicle_identification_request)) ==
299  // Wait for 2 sec to collect all the vehicle identification response
300  handler_impl_->GetSyncTimer().WaitForTimeout(
301  [&]() {
302  handler_impl_->GetStateContext().TransitionTo(
303  VehicleIdentificationState::kDoIPCtrlTimeout);
304  // Todo: Send data to upper layer here
305  },
306  [&]() {
307  // no cancellation
308  },
309  std::chrono::milliseconds{kDoIPCtrl});
310  handler_impl_->GetStateContext().TransitionTo(VehicleIdentificationState::kIdle);
311  } else {
312  // failed, do nothing
313  handler_impl_->GetStateContext().TransitionTo(VehicleIdentificationState::kIdle);
314  logger::DoipClientLogger::GetDiagClientLogger().GetLogger().LogError(
315  FILE_NAME, __LINE__, "", [](std::stringstream &msg) {
316  msg << "Vehicle Identification request transmission Failed";
317  });
318  }
319  } else {
320  // not free, state already in idle state
321  }
322  return ret_val;
323 }
324 
326  DoipMessage &doip_payload) noexcept {
327  if (handler_impl_->GetStateContext().GetActiveState().GetState() ==
328  VehicleIdentificationState::kWaitForVehicleIdentificationRes) {
329  // Deserialize data to indicate to upper layer
332  ret_val{handler_impl_->GetDoipChannel().IndicateMessage(
333  static_cast<uds_transport::UdsMessage::Address>(0U),
334  static_cast<uds_transport::UdsMessage::Address>(0U),
336  doip_payload.GetPayload().size(), 0U, "DoIPUdp", doip_payload.GetPayload())};
337  if ((ret_val.first ==
339  (ret_val.second != nullptr)) {
340  // Add meta info about ip address
342  {"kRemoteIpAddress", std::string{doip_payload.GetHostIpAddress()}}};
343  ret_val.second->AddMetaInfo(
344  std::make_shared<uds_transport::UdsMessage::MetaInfoMap>(meta_info_map));
345  // copy to application buffer
346  (void) std::copy(doip_payload.GetPayload().begin(), doip_payload.GetPayload().end(),
347  ret_val.second->GetPayload().begin());
348  handler_impl_->GetDoipChannel().HandleMessage(std::move(ret_val.second));
349  }
350  } else {
351  // ignore
352  }
353 }
354 
356  uds_transport::UdsMessageConstPtr vehicle_identification_request) noexcept
360  // Get preselection mode
361  std::uint8_t const preselection_mode{vehicle_identification_request->GetPayload()[1u]};
362  // Get the payload type & length from preselection mode
363  VehiclePayloadType const doip_vehicle_payload_type{
364  GetVehicleIdentificationPayloadType(preselection_mode)};
365 
366  // create header
367  UdpMessage::BufferType compose_vehicle_identification_req{
368  CreateDoipGenericHeader(doip_vehicle_payload_type.first, doip_vehicle_payload_type.second)};
369  // Copy only if containing VIN / EID
370  if (doip_vehicle_payload_type.first != kDoip_VehicleIdentification_ReqType) {
371  compose_vehicle_identification_req.insert(
372  compose_vehicle_identification_req.begin() + kDoipheadrSize,
373  vehicle_identification_request->GetPayload().begin() + 2U,
374  vehicle_identification_request->GetPayload().end());
375  }
376 
377  UdpMessagePtr doip_vehicle_identification_req{
378  std::make_unique<UdpMessage>(vehicle_identification_request->GetHostIpAddress(),
379  vehicle_identification_request->GetHostPortNumber(),
380  std::move(compose_vehicle_identification_req))};
381  if (handler_impl_->GetSocketHandler().Transmit(std::move(doip_vehicle_identification_req))) {
383  }
384  return ret_val;
385 }
386 
387 } // namespace udp_channel
388 } // namespace channel
389 } // namespace doip_client
Immutable class to store received doip message.
Definition: doip_message.h:21
Class to manage a udp channel as per DoIP protocol.
VehicleIdentificationHandlerImpl(sockets::UdpSocketHandler &udp_socket_handler, DoipUdpChannel &channel)
Constructs an instance of VehicleDiscoveryHandlerImpl.
auto GetStateContext() noexcept -> VehicleIdentificationStateContext &
Function to get the Vehicle Identification State context.
auto GetSocketHandler() noexcept -> sockets::UdpSocketHandler &
Function to get the socket handler.
auto SendVehicleIdentificationRequest(uds_transport::UdsMessageConstPtr vehicle_identification_request) noexcept -> uds_transport::UdsTransportProtocolMgr::TransmissionResult
Function to send vehicle identification request.
~VehicleIdentificationHandler()
Destruct an instance of VehicleIdentificationHandler.
sockets::UdpSocketHandler::MessagePtr UdpMessagePtr
Type alias for Tcp message pointer.
auto HandleVehicleIdentificationRequest(uds_transport::UdsMessageConstPtr vehicle_identification_request) noexcept -> uds_transport::UdsTransportProtocolMgr::TransmissionResult
Function to handle sending of vehicle identification request.
void ProcessVehicleIdentificationResponse(DoipMessage &doip_payload) noexcept
Function to process received vehicle identification response.
VehicleIdentificationHandler(sockets::UdpSocketHandler &udp_socket_handler, DoipUdpChannel &channel)
Constructs an instance of VehicleIdentificationHandler.
kWaitForVehicleIdentificationRes(VehicleIdentificationState state)
Constructs an instance of kWaitForVehicleIdentificationRes.
static auto GetDiagClientLogger() noexcept -> DoipClientLogger &
Definition: logger.h:20
Handler class to manage different socket of various client (Udp / Tcp)
std::map< std::string, std::string > MetaInfoMap
Definition: uds_message.h:28
void TransitionTo(EnumState state)
Definition: state.h:70
void AddState(EnumState state, std::unique_ptr< State< EnumState >> state_ptr)
Definition: state.h:58
#define FILE_NAME
Definition: file_path.h:14
std::pair< std::uint16_t, std::uint8_t > VehiclePayloadType
Type alias of vehicle payload type.
auto CreateDoipGenericHeader(std::uint16_t payload_type, std::uint32_t payload_len) noexcept -> std::vector< std::uint8_t >
Function to create doip generic header.
auto GetVehicleIdentificationPayloadType(std::uint8_t preselection_mode) noexcept -> VehiclePayloadType
Get the vehicle identification payload type based on preselection mode.
SocketHandler< boost_support::client::udp::UdpClient > UdpSocketHandler
Type alias of Udp socket handler.
constexpr std::uint8_t kDoipheadrSize
constexpr std::uint16_t kDoip_VehicleIdentification_ReqType
constexpr std::uint32_t kDoIPCtrl
constexpr std::uint32_t kDoip_VehicleIdentification_ReqLen
constexpr std::uint8_t kDoip_ProtocolVersion
constexpr std::uint32_t kDoip_VehicleIdentificationVIN_ReqLen
constexpr std::uint16_t kDoip_VehicleIdentificationVIN_ReqType
constexpr std::uint32_t kDoip_VehicleIdentificationEID_ReqLen
constexpr std::uint16_t kDoip_VehicleIdentificationEID_ReqType
std::unique_ptr< const UdsMessage > UdsMessageConstPtr
Definition: uds_message.h:69
std::unique_ptr< UdsMessage > UdsMessagePtr
Definition: uds_message.h:71
State
Definitions of different connection state.
Definition: tls_client.cpp:56