Diag-Client-Lib
dm_conversation.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 
14 
15 namespace diag {
16 namespace client {
17 namespace conversation {
18 
23  public:
32  DmConversation &dm_conversion)
33  : uds_transport::ConversionHandler{handler_id},
34  dm_conversation_{dm_conversion} {}
35 
39  DmConversationHandler(const DmConversationHandler &other) noexcept = delete;
40  DmConversationHandler &operator=(const DmConversationHandler &other) noexcept = delete;
41 
45  DmConversationHandler(DmConversationHandler &&other) noexcept = delete;
47 
51  ~DmConversationHandler() override = default;
52 
83  ::uds_transport::ChannelID channel_id, std::size_t size,
84  ::uds_transport::Priority priority, ::uds_transport::ProtocolKind protocol_kind,
85  core_type::Span<std::uint8_t const> payload_info) const noexcept override {
86  return (dm_conversation_.IndicateMessage(source_addr, target_addr, type, channel_id, size,
87  priority, protocol_kind, payload_info));
88  }
89 
96  void HandleMessage(::uds_transport::UdsMessagePtr message) const noexcept override {
97  dm_conversation_.HandleMessage(std::move(message));
98  }
99 
100  private:
105 };
106 
107 DmConversation::DmConversation(std::string_view conversion_name,
108  DMConversationType &conversion_identifier)
109  : Conversation{},
110  active_session_{SessionControlType::kDefaultSession},
111  active_security_level_{SecurityLevelType::kLocked},
112  rx_buffer_size_{conversion_identifier.rx_buffer_size},
113  p2_client_max_{conversion_identifier.p2_client_max},
114  p2_star_client_max_{conversion_identifier.p2_star_client_max},
115  source_address_{conversion_identifier.source_address},
116  target_address_{},
117  conversation_name_{conversion_name},
118  dm_conversion_handler_{
119  std::make_unique<DmConversationHandler>(conversion_identifier.handler_id, *this)} {}
120 
122 
123 void DmConversation::Startup() noexcept {
124  // initialize the connection
125  static_cast<void>(connection_->Initialize());
126  // start the connection
127  connection_->Start();
128  // Change the state to Active
130  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
131  FILE_NAME, __LINE__, __func__, [&](std::stringstream &msg) {
132  msg << "'" << conversation_name_ << "'"
133  << "-> "
134  << "Startup completed";
135  });
136 }
137 
138 void DmConversation::Shutdown() noexcept {
140  // shutdown connection
141  connection_->Stop();
142  // Change the state to InActive
144  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
145  FILE_NAME, __LINE__, __func__, [&](std::stringstream &msg) {
146  msg << "'" << conversation_name_ << "'"
147  << "-> "
148  << "Shutdown completed";
149  });
150  }
151 }
152 
154  std::uint16_t const target_address, IpAddress const host_ip_addr) noexcept {
155  // create an uds message just to get the port number
156  // source address required for Routing Activation
157  uds_transport::ByteVector payload{}; // empty payload
158  // Send Connect request to doip layer
159  DiagClientConversation::ConnectResult const connection_result{
161  connection_->ConnectToHost(std::make_unique<diag::client::uds_message::DmUdsMessage>(
162  source_address_, target_address, host_ip_addr, payload)))};
163  remote_address_ = host_ip_addr;
164  target_address_ = target_address;
166  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
167  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
168  msg << "'" << conversation_name_ << "'"
169  << "-> "
170  << "Successfully connected to Server with IP <" << remote_address_ << ">"
171  << " and LA= 0x" << std::hex << target_address_;
172  });
173  } else {
174  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogError(
175  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
176  msg << "'" << conversation_name_ << "'"
177  << "-> "
178  << "Failed connecting to Server with IP <" << remote_address_ << ">"
179  << " and LA= 0x" << std::hex << target_address_;
180  });
181  }
182  return connection_result;
183 }
184 
188  // Check if already connected before disconnecting
189  if (connection_->IsConnectToHost()) {
190  // Send disconnect request to doip layer
191  ret_val =
192  static_cast<DiagClientConversation::DisconnectResult>(connection_->DisconnectFromHost());
194  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
195  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
196  msg << "'" << conversation_name_ << "'"
197  << "-> "
198  << "Successfully disconnected from Server with IP <" << remote_address_ << ">"
199  << " and LA= 0x" << std::hex << target_address_;
200  });
201  } else {
202  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogWarn(
203  FILE_NAME, __LINE__, __func__, [&](std::stringstream &msg) {
204  msg << "'" << conversation_name_ << "'"
205  << "-> "
206  << "Failed to disconnect from Server with IP <" << remote_address_ << ">";
207  });
208  }
209  } else {
211  }
212  return ret_val;
213 }
214 
220  if (message) {
221  // fill the data
222  uds_transport::ByteVector payload{message->GetPayload()};
223  // Initiate Sending of diagnostic request
225  connection_->Transmit(std::make_unique<diag::client::uds_message::DmUdsMessage>(
226  source_address_, target_address_, message->GetHostIpAddress(), payload))};
227  if (transmission_result ==
229  // Diagnostic Request Sent successful
230  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
231  FILE_NAME, __LINE__, __func__, [&](std::stringstream &msg) {
232  msg << "'" << conversation_name_ << "'"
233  << "-> "
234  << "Diagnostic Request Sent & Positive Ack received";
235  });
236  conversation_state_.GetConversationStateContext().TransitionTo(
237  ConversationState::kDiagWaitForRes);
238  // Wait P6Max / P2ClientMax
239  sync_timer_.WaitForTimeout(
240  [this, &result]() {
242  conversation_state_.GetConversationStateContext().TransitionTo(
243  ConversationState::kIdle);
244  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
245  FILE_NAME, __LINE__, "", [&](std::stringstream &msg) {
246  msg << "'" << conversation_name_ << "'"
247  << "-> "
248  << "Diagnostic Response P2 Timeout happened after " << p2_client_max_
249  << " milliseconds";
250  });
251  },
252  [this]() {
253  // pending or pos/neg response
254  if (conversation_state_.GetConversationStateContext().GetActiveState().GetState() ==
255  ConversationState::kDiagRecvdFinalRes) {
256  // pos/neg response received
257  } else if (conversation_state_.GetConversationStateContext()
258  .GetActiveState()
259  .GetState() == ConversationState::kDiagRecvdPendingRes) {
260  // first pending received
261  conversation_state_.GetConversationStateContext().TransitionTo(
262  ConversationState::kDiagStartP2StarTimer);
263  }
264  },
265  std::chrono::milliseconds{p2_client_max_});
266 
267  // Wait until final response or timeout
268  while (conversation_state_.GetConversationStateContext().GetActiveState().GetState() !=
269  ConversationState::kIdle) {
270  // Check the active state
271  switch (conversation_state_.GetConversationStateContext().GetActiveState().GetState()) {
272  case ConversationState::kDiagRecvdPendingRes:
273  conversation_state_.GetConversationStateContext().TransitionTo(
274  ConversationState::kDiagStartP2StarTimer);
275  break;
276  case ConversationState::kDiagRecvdFinalRes:
277  // do nothing
278  break;
279  case ConversationState::kDiagStartP2StarTimer:
280  // wait P6Star/ P2 star client time
281  sync_timer_.WaitForTimeout(
282  [this, &result]() {
283  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
284  FILE_NAME, __LINE__, "", [&](std::stringstream &msg) {
285  msg << "'" << conversation_name_ << "'"
286  << "-> "
287  << "Diagnostic Response P2 Star Timeout "
288  "happened after "
289  << p2_star_client_max_ << " milliseconds";
290  });
292  conversation_state_.GetConversationStateContext().TransitionTo(
293  ConversationState::kIdle);
294  },
295  [this]() {
296  // pending or pos/neg response
297  if (conversation_state_.GetConversationStateContext()
298  .GetActiveState()
299  .GetState() == ConversationState::kDiagRecvdFinalRes) {
300  // pos/neg response received
301  } else if (conversation_state_.GetConversationStateContext()
302  .GetActiveState()
303  .GetState() == ConversationState::kDiagRecvdPendingRes) {
304  // pending received again
305  conversation_state_.GetConversationStateContext().TransitionTo(
306  ConversationState::kDiagStartP2StarTimer);
307  }
308  },
309  std::chrono::milliseconds{p2_star_client_max_});
310  break;
311  case ConversationState::kDiagSuccess:
312  // change state to idle, form the uds response and return
313  result.EmplaceValue(
314  std::make_unique<diag::client::uds_message::DmUdsResponse>(payload_rx_buffer_));
315  conversation_state_.GetConversationStateContext().TransitionTo(
316  ConversationState::kIdle);
317  break;
318  default:
319  // nothing
320  break;
321  }
322  }
323  } else {
324  // failure
325  result.EmplaceError(ConvertResponseType(transmission_result));
326  }
327  } else {
329  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogWarn(
330  FILE_NAME, __LINE__, "", [&](std::stringstream &msg) {
331  msg << "'" << conversation_name_ << "'"
332  << "-> "
333  << "Diagnostic Request message is empty";
334  });
335  }
336  return result;
337 }
338 
340  std::unique_ptr<uds_transport::Connection> connection) noexcept {
341  connection_ = std::move(connection);
342 }
343 
345  return *dm_conversion_handler_;
346 }
347 
348 std::pair<uds_transport::UdsTransportProtocolMgr::IndicationResult, uds_transport::UdsMessagePtr>
354  core_type::Span<std::uint8_t const> payload_info) noexcept {
355  std::pair<uds_transport::UdsTransportProtocolMgr::IndicationResult, uds_transport::UdsMessagePtr>
357  // Verify the payload received :-
358  if (!payload_info.empty()) {
359  // Check for size, else kIndicationOverflow
360  if (size <= rx_buffer_size_) {
361  // Check for pending response
362  // payload = 0x7F XX 0x78
363  if (payload_info[0U] == 0x7F && payload_info[2U] == 0x78) {
364  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
365  FILE_NAME, __LINE__, "", [&](std::stringstream &msg) {
366  msg << "'" << conversation_name_ << "'"
367  << "-> "
368  << "Diagnostic pending response received in Conversation";
369  });
370  ret_val.first =
372  conversation_state_.GetConversationStateContext().TransitionTo(
373  ConversationState::kDiagRecvdPendingRes);
374  } else {
375  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogDebug(
376  FILE_NAME, __LINE__, "", [this](std::stringstream &msg) {
377  msg << "'" << conversation_name_ << "'"
378  << "-> "
379  << "Diagnostic final response received in Conversation";
380  });
381  // positive or negative response, provide valid buffer
382  // resize the global rx buffer
383  payload_rx_buffer_.resize(size);
385  ret_val.second = std::make_unique<diag::client::uds_message::DmUdsMessage>(
386  source_address_, target_address_, "", payload_rx_buffer_);
387  conversation_state_.GetConversationStateContext().TransitionTo(
388  ConversationState::kDiagRecvdFinalRes);
389  }
390  sync_timer_.CancelWait();
391  } else {
392  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogError(
393  FILE_NAME, __LINE__, "", [&](std::stringstream &msg) {
394  msg << "'" << conversation_name_ << "'"
395  << "-> "
396  << "Diagnostic Conversation Error Indication Overflow";
397  });
399  }
400  } else {
401  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogError(
402  FILE_NAME, __LINE__, "", [&](std::stringstream &msg) {
403  msg << "'" << conversation_name_ << "'"
404  << "-> "
405  << "Diagnostic Conversation Rx Payload size 0 received";
406  });
407  }
408  return ret_val;
409 }
410 
412  if (message != nullptr) {
413  conversation_state_.GetConversationStateContext().TransitionTo(ConversationState::kDiagSuccess);
414  }
415 }
416 
421  switch (result_type) {
424  break;
427  break;
430  break;
433  break;
434  default:
436  break;
437  }
438  return ret_result;
439 }
440 
441 } // namespace conversation
442 } // namespace client
443 } // namespace diag
Class type to contains a value (of type ValueType), or an error (of type ErrorType)
Definition: result.h:29
static Result FromError(const E &e) noexcept
Build a new Result from the specified error (given as lvalue)
Definition: result.h:83
A view over a contiguous sequence of objects.
Definition: span.h:191
Interface for diag client conversation.
Definition: conversation.h:24
DiagClientConversation::IpAddress IpAddress
Type alias for Ip address.
Definition: conversation.h:29
auto GetActivityStatus() const noexcept -> ActivityStatusType
Get the current activity status of this conversation.
Definition: conversation.h:196
ActivityStatusType activity_status_
Store the conversation activity status.
Definition: conversation.h:202
DiagError
Definitions of Diagnostics Request Response results.
Class to manage reception from transport protocol handler to dm connection handler.
~DmConversationHandler() override=default
Destructs an instance of DmConversationHandler.
DmConversationHandler(::uds_transport::conversion_manager::ConversionHandlerID handler_id, DmConversation &dm_conversion)
Constructs an instance of DmConversationHandler.
DmConversationHandler & operator=(const DmConversationHandler &other) noexcept=delete
DmConversation & dm_conversation_
Store the reference of dm conversation.
std::pair<::uds_transport::UdsTransportProtocolMgr::IndicationResult, ::uds_transport::UdsMessagePtr > IndicateMessage(::uds_transport::UdsMessage::Address source_addr, ::uds_transport::UdsMessage::Address target_addr, ::uds_transport::UdsMessage::TargetAddressType type, ::uds_transport::ChannelID channel_id, std::size_t size, ::uds_transport::Priority priority, ::uds_transport::ProtocolKind protocol_kind, core_type::Span< std::uint8_t const > payload_info) const noexcept override
Function to indicate a start of reception of message.
DmConversationHandler & operator=(DmConversationHandler &&other) noexcept=delete
void HandleMessage(::uds_transport::UdsMessagePtr message) const noexcept override
Function to Hands over a valid received Uds message.
DmConversationHandler(DmConversationHandler &&other) noexcept=delete
Deleted move assignment and move constructor.
DmConversationHandler(const DmConversationHandler &other) noexcept=delete
Deleted copy assignment and copy constructor.
Class to establish connection with Diagnostic Server.
void Startup() noexcept override
Function to start the DmConversation.
~DmConversation() override
Destructs an instance of DmConversation.
std::pair<::uds_transport::UdsTransportProtocolMgr::IndicationResult, ::uds_transport::UdsMessagePtr > IndicateMessage(::uds_transport::UdsMessage::Address source_addr, ::uds_transport::UdsMessage::Address target_addr, ::uds_transport::UdsMessage::TargetAddressType type, ::uds_transport::ChannelID channel_id, std::size_t size, ::uds_transport::Priority priority, ::uds_transport::ProtocolKind protocol_kind, core_type::Span< std::uint8_t const > payload_info) noexcept override
Function to indicate a start of reception of message.
DisconnectResult DisconnectFromDiagServer() noexcept override
Function to disconnect from Diagnostic Server.
std::unique_ptr<::uds_transport::ConversionHandler > dm_conversion_handler_
Store the dm conversation handler.
DmConversation(std::string_view conversion_name, DMConversationType &conversion_identifier)
Constructs an instance of DmConversation.
void HandleMessage(::uds_transport::UdsMessagePtr message) noexcept override
Function to Hands over a valid received Uds message.
void RegisterConnection(std::unique_ptr<::uds_transport::Connection > connection) noexcept override
Function to register the conversation to underlying transport protocol handler.
ConnectResult ConnectToDiagServer(std::uint16_t target_address, IpAddress host_ip_addr) noexcept override
Function to connect to Diagnostic Server.
SecurityLevelType
Definitions of active security level.
::uds_transport::ConversionHandler & GetConversationHandler() noexcept override
Function to get the conversation handler from conversation object.
Result< uds_message::UdsResponseMessagePtr, DiagError > SendDiagnosticRequest(uds_message::UdsRequestMessageConstPtr message) noexcept override
Function to send Diagnostic Request and get Diagnostic Response.
std::uint16_t target_address_
Store the logical target address of remote server.
std::string remote_address_
Store the remote IP address of remote server.
void Shutdown() noexcept override
Function to shutdown the DmConversation.
std::unique_ptr<::uds_transport::Connection > connection_
Store the underlying transport protocol connection object.
std::string conversation_name_
Store the conversation name.
SessionControlType
Definitions of active diagnostic session.
static DiagClientConversation::DiagError ConvertResponseType(::uds_transport::UdsTransportProtocolMgr::TransmissionResult result_type)
Helper function to convert response type.
static auto GetDiagClientLogger() noexcept -> DiagClientLogger &
Get the diag client logger instance.
Definition: logger.h:32
Class to manage reception from transport protocol handler to connection handler.
ConversionHandler(conversion_manager::ConversionHandlerID handler_id)
Constructs an instance of ConversionHandler.
#define FILE_NAME
Definition: file_path.h:14
std::unique_ptr< const UdsMessage > UdsRequestMessageConstPtr
Type alias of unique_ptr for constant UdsRequestMessage.
std::string_view ProtocolKind
std::unique_ptr< UdsMessage > UdsMessagePtr
Definition: uds_message.h:71
uint32_t ChannelID
std::uint8_t Priority
std::vector< std::uint8_t > ByteVector
Structure containing DM conversation type.