SVHReceiveThread.cpp
Go to the documentation of this file.
1 //
3 // © Copyright 2022 SCHUNK Mobile Greifsysteme GmbH, Lauffen/Neckar Germany
4 // © Copyright 2022 FZI Forschungszentrum Informatik, Karlsruhe, Germany
5 //
6 // This file is part of the Schunk SVH Library.
7 //
8 // The Schunk SVH Library is free software: you can redistribute it and/or
9 // modify it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation, either version 3 of the License, or (at your
11 // option) any later version.
12 //
13 // The Schunk SVH Library is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
16 // Public License for more details.
17 //
18 // You should have received a copy of the GNU General Public License along with
19 // the Schunk SVH Library. If not, see <https://www.gnu.org/licenses/>.
20 //
22 
23 //----------------------------------------------------------------------
36 //----------------------------------------------------------------------
37 #include <chrono>
40 #include <sstream>
41 #include <thread>
42 
44 
45 namespace driver_svh {
46 
47 SVHReceiveThread::SVHReceiveThread(const std::chrono::microseconds& idle_sleep,
48  std::shared_ptr<Serial> device,
49  ReceivedPacketCallback const& received_callback)
50  : m_idle_sleep(idle_sleep)
51  , m_serial_device(device)
52  , m_received_state(RS_HEADE_R1)
53  , m_length(0)
54  , m_data(0, 0)
55  , m_ab(0)
56  , m_packets_received(0)
57  , m_skipped_bytes(0)
58  , m_received_callback(received_callback)
59 {
60 }
61 
63 {
64  while (m_continue)
65  {
66  if (m_serial_device) // != NULL)
67  {
68  if (m_serial_device->isOpen())
69  {
70  auto start = std::chrono::high_resolution_clock::now();
71 
72  // All we every want to do is receiving data :)
73  if (!receiveData())
74  {
75  auto elapsed_time = std::chrono::high_resolution_clock::now() - start;
76 
77  if ((m_idle_sleep - elapsed_time).count() > 0) // sleep remainder of the cycle
78  {
79  std::this_thread::sleep_for(m_idle_sleep - elapsed_time);
80  }
81  else // We exceeded at least one cycle time. Sleep until we are back in sync.
82  {
83  std::this_thread::sleep_for(elapsed_time % m_idle_sleep);
84  }
85  }
86  }
87  else
88  {
89  SVH_LOG_WARN_STREAM("SVHReceiveThread",
90  "Cannot read data from serial device. It is not opened!");
91  std::this_thread::sleep_for(m_idle_sleep); // we can neglect the processing time to get here
92  }
93  }
94  else
95  {
96  // Wait for the thread period so that the timing is in sync.
97  std::this_thread::sleep_for(m_idle_sleep); // we can neglect the processing time to get here
98  }
99  }
100 }
101 
103 {
104  /*
105  * Each packet has to follow the defined packet structure which is ensured by the following state
106  * machine. The "Bytestream" (not realy a stream) is interpreted byte by byte. If the structure is
107  * still right the next state is entered, if a wrong byte is detected the whole packet is
108  * discarded and the SM switches to the synchronization state aggain. If the SM reaches the final
109  * state the packet will be given to the packet handler to decide what to do with its content.
110  * NOTE: All layers working with a SerialPacket (except this one) assume that the packet has a
111  * valid structure and all data fields present.
112  */
113  uint8_t data_byte;
114  ssize_t bytes = m_serial_device->read(&data_byte, sizeof(uint8_t));
115  if (bytes < 0)
116  {
117  SVH_LOG_DEBUG_STREAM("SVHReceiveThread", "Serial read error:" << bytes);
118  return false;
119  }
120  if (bytes < 1)
121  {
122  return false;
123  }
124 
125  switch (m_received_state)
126  {
127  case RS_HEADE_R1: {
128  if (data_byte == PACKET_HEADER1)
129  {
131  }
132  else
133  {
134  m_skipped_bytes++;
135  }
136  break;
137  }
138  case RS_HEADE_R2: {
139  switch (data_byte)
140  {
141  case PACKET_HEADER2: {
143  break;
144  }
145  case PACKET_HEADER1: {
147  m_skipped_bytes++;
148  break;
149  }
150  default: {
152  m_skipped_bytes += 2;
153  break;
154  }
155  }
156  break;
157  }
158  case RS_INDEX: {
159  // Reset Array Builder for each fresh packet
160  m_ab.reset(0);
161 
162  // Data bytes are not cenverted in endianess at this point
163  m_ab.appendWithoutConversion(data_byte);
165  break;
166  }
167  case RS_ADDRESS: {
168  // get the address
169  m_ab.appendWithoutConversion(data_byte);
171  break;
172  }
173  case RS_LENGT_H1: {
174  // get payload length
175  m_ab.appendWithoutConversion(data_byte);
177  break;
178  }
179  case RS_LENGT_H2: {
180  // get payload length
181  m_ab.appendWithoutConversion(data_byte);
182  m_length = m_ab.readBack<uint16_t>();
184  m_data.clear();
185  m_data.reserve(m_length);
186  break;
187  }
188  case RS_DATA: {
189  // get the payload itself
190  // Some conversion due to legacy hardware calls
191  m_data.push_back(data_byte);
192  m_ab.appendWithoutConversion(data_byte);
193  if (m_data.size() >= m_length)
194  {
196  }
197  break;
198  }
199  case RS_CHECKSU_M1: {
200  m_checksum1 = data_byte;
201  m_checksum2 = 0;
203  break;
204  }
205  case RS_CHECKSU_M2: {
206  m_checksum2 = data_byte;
207  uint8_t checksum1 = m_checksum1;
208  uint8_t checksum2 = m_checksum2;
209  // probe for correct checksum
210  for (size_t i = 0; i < m_data.size(); ++i)
211  {
212  checksum1 -= m_data[i];
213  checksum2 ^= m_data[i];
214  }
215 
216  if ((checksum1 == 0) && (checksum2 == 0))
217  {
218  // start with an empty package
219  // Warning: It is imperative for correct readouts to create the received_packet with the
220  // correct length!
221  SVHSerialPacket received_packet(m_length);
222  m_ab >> received_packet;
223 
225 
226  if (m_skipped_bytes > 0)
227  SVH_LOG_DEBUG_STREAM("SVHReceiveThread", "Skipped " << m_skipped_bytes << " bytes ");
228  SVH_LOG_DEBUG_STREAM("SVHReceiveThread",
229  "Received packet index:" << received_packet.index
230  << ", address:" << received_packet.address
231  << ", size:" << received_packet.data.size());
232  m_skipped_bytes = 0;
233 
234  // notify whoever is waiting for this
236  {
237  m_received_callback(received_packet, m_packets_received);
238  }
239 
241  }
242  else
243  {
245 
246  SVHSerialPacket received_packet(m_length);
247  m_ab >> received_packet;
248 
249  if (m_skipped_bytes > 0)
250  SVH_LOG_DEBUG_STREAM("SVHReceiveThread", "Skipped " << m_skipped_bytes << " bytes: ");
251  SVH_LOG_DEBUG_STREAM("SVHReceiveThread",
252  "Checksum error: " << (int)checksum1 << "," << (int)checksum2
253  << "!=0, skipping " << m_length + 8
254  << "bytes, packet index:" << received_packet.index
255  << ", address:" << received_packet.address
256  << ", size:" << received_packet.data.size());
257  m_skipped_bytes = 0;
259  {
260  m_received_callback(received_packet, m_packets_received);
261  }
262  }
263  break;
264  }
265  }
266 
267  return true;
268 }
269 
270 } // namespace driver_svh
std::vector< uint8_t > m_data
length of received serial data
const uint8_t PACKET_HEADER1
Header sync byte 1.
tState m_received_state
current state of the state machine
unsigned int m_skipped_bytes
counter for skipped bytes in case no packet is detected
const uint8_t PACKET_HEADER2
Header sync byte 2.
ReceivedPacketCallback m_received_callback
function callback for received packages
bool receiveData()
state machine processing received data
void run()
run method of the thread, executes the main program in an infinite loop
uint16_t m_length
length of received serial data
std::function< void(const SVHSerialPacket &packet, unsigned int packet_count)> ReceivedPacketCallback
definition of function callback for received packages
std::shared_ptr< Serial > m_serial_device
pointer to serial device object
#define SVH_LOG_DEBUG_STREAM(NAME, M)
Definition: Logger.h:39
driver_svh::ArrayBuilder m_ab
pointer to array builder object for packet receive
void appendWithoutConversion(const T &data)
add data without any byte conversion
std::chrono::microseconds m_idle_sleep
sleep time during run() if idle
std::atomic< bool > m_continue
Flag to end the run() method from external callers.
void reset(size_t array_size=1)
Resets the Arraybuilder to initial state, all values will be deleted.
uint8_t m_checksum1
Checksum of packet.
SVHReceiveThread(const std::chrono::microseconds &idle_sleep, std::shared_ptr< Serial > device, ReceivedPacketCallback const &received_callback)
SVHReceiveThread Constructs a new Receivethread.
#define SVH_LOG_WARN_STREAM(NAME, M)
Definition: Logger.h:53
std::atomic< unsigned int > m_packets_received
packets counter
The SerialPacket holds the (non generated) header and data of one message to the SVH-Hardware.


schunk_svh_library
Author(s): Georg Heppner, Lars Pfotzer, Felix Exner, Johannes Mangler, Stefan Scherzinger, Pascal Becker
autogenerated on Fri Apr 14 2023 02:26:23