• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2021 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 #ifndef HDC_UART_H
16 #define HDC_UART_H
17 #include "common.h"
18 
19 #include <chrono>
20 #include <numeric>
21 #include <sstream>
22 #include <unordered_set>
23 
24 #ifndef _WIN32
25 #include <termios.h> // struct termios
26 #endif               // _WIN32
27 
28 namespace Hdc {
29 #define USE_UART_CHECKSUM // all the data and head will have a checksum
30 #undef HDC_UART_TIMER_LOG // will have a lot of log from timer
31 
32 enum UartProtocolOption {
33     PKG_OPTION_TAIL = 1,  // makr is the last packget, can be send to session.
34     PKG_OPTION_RESET = 2, // host request reset session in daemon
35     PKG_OPTION_ACK = 4,   // reponse the pkg is received
36     PKG_OPTION_NAK = 8,   // requeset resend pkg again
37     PKG_OPTION_FREE = 16, // request free this session, some unable recovery error happend
38 };
39 
40 static_assert(MAX_UART_SIZE_IOBUF != 0);
41 
42 #pragma pack(push)
43 #pragma pack(1)
44 struct UartHead {
45     UartHead(const UartHead &) = delete;
46     UartHead &operator=(const UartHead &) = delete;
47     UartHead(UartHead &&) = default;
48 
49     uint8_t flag[2];           // magic word
50     uint16_t option;           // UartProtocolOption
51     uint32_t sessionId;        // the package owner (COM dev owner)
52     uint32_t dataSize;         // data size not include head
53     uint32_t packageIndex;     // package index in this session
54     uint32_t dataCheckSum = 0; // data checksum
55     uint32_t headCheckSum = 0; // head checksum
56     std::string ToPkgIdentityString(bool responsePackage = false) const
57     {
58         std::ostringstream oss;
59         if (responsePackage) {
60             oss << "R-";
61         }
62         oss << "Id:" << sessionId;
63         oss << "pkgIdx:" << packageIndex;
64         return oss.str();
65     };
ToDebugStringUartHead66     std::string ToDebugString() const
67     {
68         std::ostringstream oss;
69         oss << "UartHead [";
70         oss << " flag:" << std::hex << unsigned(flag[0]) << " " << unsigned(flag[1]) << std::dec;
71         oss << " option:" << unsigned(option);
72         oss << " sessionId:" << sessionId;
73         oss << " dataSize:" << dataSize;
74         oss << " packageIndex:" << packageIndex;
75         if (dataSize != 0) {
76             oss << " dataCheckSum:" << std::hex << dataCheckSum;
77         }
78         oss << " headCheckSum:" << std::hex << headCheckSum;
79         oss << std::dec;
80         oss << "]";
81         return oss.str();
82     };
83     UartHead(uint32_t sessionIdIn = 0, uint8_t optionIn = 0, uint32_t dataSizeIn = 0,
84              uint32_t packageIndexIn = 0)
85         : flag {PACKET_FLAG[0], PACKET_FLAG[1]},
86           option(optionIn),
87           sessionId(sessionIdIn),
88           dataSize(dataSizeIn),
89           packageIndex(packageIndexIn)
90     {
91     }
92     bool operator==(const UartHead &r) const
93     {
94         return flag[0] == r.flag[0] and flag[1] == r.flag[1] and option == r.option and
95                dataSize == r.dataSize and packageIndex == r.packageIndex;
96     }
IsResponsePackageUartHead97     bool IsResponsePackage() const
98     {
99         return (option & PKG_OPTION_ACK) or (option & PKG_OPTION_NAK);
100     }
UpdateCheckSumUartHead101     void UpdateCheckSum()
102     {
103 #ifdef USE_UART_CHECKSUM
104         if (dataSize != 0) {
105             const uint8_t *data = reinterpret_cast<const uint8_t *>(this) + sizeof(UartHead);
106             dataCheckSum = std::accumulate(data, data + dataSize, 0u);
107         }
108         const uint8_t *head = reinterpret_cast<const uint8_t *>(this);
109         size_t headCheckSumLen = sizeof(UartHead) - sizeof(headCheckSum);
110         headCheckSum = std::accumulate(head, head + headCheckSumLen, 0u);
111 #endif
112     }
ValidateHeadUartHead113     bool ValidateHead() const
114     {
115 #ifdef USE_UART_CHECKSUM
116         const uint8_t *head = reinterpret_cast<const uint8_t *>(this);
117         size_t headCheckSumLen = sizeof(UartHead) - sizeof(headCheckSum);
118         return (headCheckSum == std::accumulate(head, head + headCheckSumLen, 0u));
119 #else
120         return true;
121 #endif
122     }
ValidateDataUartHead123     bool ValidateData() const
124     {
125 #ifdef USE_UART_CHECKSUM
126         const uint8_t *data = reinterpret_cast<const uint8_t *>(this) + sizeof(UartHead);
127         if (dataSize == 0) {
128             return true;
129         } else {
130             return (dataCheckSum == std::accumulate(data, data + dataSize, 0u));
131         }
132 #else
133         return true;
134 #endif
135     }
136 };
137 #pragma pack(pop)
138 
139 // we need virtual interface for UT the free function
140 class ExternInterface {
141 public:
142     virtual void SetTcpOptions(uv_tcp_t *tcpHandle);
143     virtual int SendToStream(uv_stream_t *handleStream, const uint8_t *buf, const int bufLen);
144     virtual int UvTcpInit(uv_loop_t *, uv_tcp_t *, int);
145     virtual int UvRead(uv_stream_t *, uv_alloc_cb, uv_read_cb);
146     virtual int StartWorkThread(uv_loop_t *loop, uv_work_cb pFuncWorkThread,
147                                 uv_after_work_cb pFuncAfterThread, void *pThreadData);
148     virtual void TryCloseHandle(const uv_handle_t *handle, uv_close_cb closeCallBack = nullptr);
149     virtual bool TimerUvTask(uv_loop_t *loop, void *data, uv_timer_cb cb);
150     virtual bool UvTimerStart(uv_timer_t *handle, uv_timer_cb cb, uint64_t timeout,
151                               uint64_t repeat);
152     using DelayCB = std::function<void(const uint8_t, string &, const void *)>;
153     virtual bool DelayDo(uv_loop_t *loop, const int delayMs, const uint8_t flag, string msg,
154                          void *data, DelayCB cb);
155     virtual ~ExternInterface() = default;
156 };
157 class HdcSessionBase;
158 class HdcUARTBase {
159 public:
160     static ExternInterface defaultInterface;
161     HdcUARTBase(HdcSessionBase &, ExternInterface & = defaultInterface);
162     virtual ~HdcUARTBase();
163     bool ReadyForWorkThread(HSession hSession);
164     int SendUARTData(HSession hSession, uint8_t *data, const size_t length);
165     // call from session side
166     // we need know when we need clear the pending send data
167     virtual void StopSession(HSession hSession);
168 
169 protected:
170     static constexpr uint32_t DEFAULT_BAUD_RATE_VALUE = 921600;
171 
172     bool stopped = false; // stop only can be call one times
173 
174     // something is processing on working thread
175     // Mainly used to reply a data back before stop.
176     std::mutex workThreadProcessingData;
177 
178     // review how about make a HUART in daemon side and put the devhandle in it ?
179     int uartHandle = -1;
180     virtual bool SendUARTRaw(HSession hSession, uint8_t *data, const size_t length);
SendUartSoftReset(HSession hUART,uint32_t sessionId)181     virtual void SendUartSoftReset(HSession hUART, uint32_t sessionId) {};
182     virtual RetErrCode ValidateUartPacket(vector<uint8_t> &data, uint32_t &sessionId,
183                                           uint32_t &packageIndex, size_t &fullPackageLength);
184     virtual void NotifyTransfer();
ResetOldSession(uint32_t sessionId)185     virtual void ResetOldSession(uint32_t sessionId)
186     {
187         return;
188     }
189     virtual void Restartession(const HSession session);
190 
191 #ifndef _WIN32
192     int SetSerial(int fd, int nSpeed, int nBits, char nEvent, int nStop);
193 #endif // _WIN32
194     virtual bool UartSendToHdcStream(HSession hSession, uint8_t *data, size_t size);
195     static void ReadDataFromUARTStream(uv_stream_t *stream, ssize_t nread, const uv_buf_t *buf);
196     bool uartOpened;
197 
198     static constexpr size_t MAX_READ_BUFFER = MAX_UART_SIZE_IOBUF * 10;
199     static constexpr int ReadGiveUpTimeOutTimeMs = 500; // 500ms
200     virtual int UartToHdcProtocol(uv_stream_t *stream, uint8_t *appendData, int dataSize);
201     int GetUartSpeed(int speed);
202     int GetUartBits(int bits);
203     virtual void ResponseUartTrans(uint32_t sessionId, uint32_t packageIndex,
204                                    UartProtocolOption option);
205 
206     virtual size_t PackageProcess(vector<uint8_t> &data, HSession hSession = nullptr);
207     virtual RetErrCode DispatchToWorkThread(HSession hSession, uint8_t *readBuf, int readBytes);
208 
209     virtual void OnTransferError(const HSession session) = 0;
210     virtual HSession GetSession(const uint32_t sessionId, bool create = false) = 0;
211 
212     /*
213         read data from uart devices
214         Args:
215         readBuf         data will append to readBuf
216         expectedSize    function will not return until expected size read
217 
218         Return:
219         ssize_t         >   0 how many bytes read after this function called
220                         ==  0 nothing read , timeout happend(expectedSize > 0)
221                         <   0 means devices error
222     */
223 
224     // we have some oswait in huart(bind to each session/uart device)
225     virtual ssize_t ReadUartDev(std::vector<uint8_t> &readBuf, size_t expectedSize, HdcUART &uart);
226 
227     virtual ssize_t WriteUartDev(uint8_t *data, const size_t length, HdcUART &uart);
228 
229     ExternInterface &externInterface;
230 
231     virtual void RequestSendPackage(uint8_t *data, const size_t length, bool queue = true);
232     virtual void ProcessResponsePackage(const UartHead &head);
233     virtual void SendPkgInUARTOutMap();
234     virtual void ClearUARTOutMap(uint32_t sessionId);
235     virtual void EnsureAllPkgsSent();
236     static constexpr int WaitResponseTimeOutMs = 1000; // 1000ms
237     static constexpr int OneMoreMs = 1;
238 
239     class TransferStateMachine {
240     public:
Request()241         void Request()
242         {
243             std::unique_lock<std::mutex> lock(mutex);
244             requested = true;
245             cv.notify_one();
246         }
247 
Sent()248         void Sent()
249         {
250             std::unique_lock<std::mutex> lock(mutex);
251             timeout = true;
252             // wait_for will timeout in 999ms in linux platform, so we add one more
253             timeoutPoint = std::chrono::steady_clock::now() +
254                            std::chrono::milliseconds(WaitResponseTimeOutMs + OneMoreMs);
255             cv.notify_one();
256         }
257 
258         void Wait();
259 
260     private:
261         std::mutex mutex;
262         std::condition_variable cv;
263         bool requested = false; // some one request send something
264         std::chrono::steady_clock::time_point timeoutPoint;
265         bool timeout = false; // some data is sendout and next wait need wait response
266     } transfer;
267 
268 private:
269     HdcSessionBase &sessionBase;
270 
271     enum PkgStatus {
272         PKG_WAIT_SEND,
273         PKG_WAIT_RESPONSE,
274     };
275     struct HandleOutputPkg {
276         std::string key;
277         uint32_t sessionId = 0; // like group , sometimes we will delete by this filter
278         bool response;          // PKG for response
279         bool ack;               // UartResponseCode for this packge
280         uint8_t pkgStatus;
281         vector<uint8_t> msgSendBuf;
282         size_t retryChance = 4; // how many time need retry
283         std::chrono::time_point<std::chrono::steady_clock> sendTimePoint;
284         // reivew if we need direct process UartHead ?
285         HandleOutputPkg(std::string keyIn, uint32_t sessionIdIn, uint8_t *data, size_t length,
286                         bool responseIn = false, bool ackIn = false)
keyHandleOutputPkg287             : key(keyIn),
288               sessionId(sessionIdIn),
289               response(responseIn),
290               ack(ackIn),
291               pkgStatus(PKG_WAIT_SEND),
292               msgSendBuf(data, data + length)
293         {
294         }
ToDebugStringHandleOutputPkg295         std::string ToDebugString()
296         {
297             std::string debug;
298             debug.append(key);
299             debug.append(" pkgStatus:");
300             debug.append(std::to_string(pkgStatus));
301             if (pkgStatus == PKG_WAIT_RESPONSE) {
302                 debug.append(" sent:");
303                 auto elapsedTime = std::chrono::duration_cast<std::chrono::milliseconds>(
304                     std::chrono::steady_clock::now() - sendTimePoint);
305                 debug.append(std::to_string(elapsedTime.count()));
306                 debug.append(" ms");
307                 debug.append(" retry Chance:");
308                 debug.append(std::to_string(retryChance));
309             }
310             if (response) {
311                 debug.append(" response:");
312                 if (ack) {
313                     debug.append(" ACK");
314                 } else {
315                     debug.append(" NAK");
316                 }
317             }
318             return debug;
319         }
320     };
321 
322     class TransferSlot {
323     public:
Wait(uint32_t sessionId)324         void Wait(uint32_t sessionId)
325         {
326             std::unique_lock<std::mutex> lock(mutex);
327             cv.wait(lock, [=] { return hasWaitPkg.find(sessionId) == hasWaitPkg.end(); });
328             hasWaitPkg.emplace(sessionId);
329         }
330 
Free(uint32_t sessionId)331         void Free(uint32_t sessionId)
332         {
333             std::unique_lock<std::mutex> lock(mutex);
334             hasWaitPkg.erase(sessionId);
335             cv.notify_one();
336         }
337 
338         // call when exit
WaitFree()339         void WaitFree()
340         {
341             std::unique_lock<std::mutex> lock(mutex);
342             cv.wait_for(lock, std::chrono::milliseconds(WaitResponseTimeOutMs),
343                         [=] { return hasWaitPkg.size() == 0; });
344         }
345 
346     private:
347         std::mutex mutex;
348         std::condition_variable cv;
349         std::unordered_set<uint32_t> hasWaitPkg;
350     } slots;
351 
352     vector<HandleOutputPkg> outPkgs; // Pkg label, HOutPkg
353     std::recursive_mutex mapOutPkgsMutex;
354     struct HandleOutputPkgKeyFinder {
355         const std::string &key;
HandleOutputPkgKeyFinderHandleOutputPkgKeyFinder356         HandleOutputPkgKeyFinder(const std::string &keyIn) : key(keyIn) {}
operatorHandleOutputPkgKeyFinder357         bool operator()(const HandleOutputPkg &other)
358         {
359             return key == other.key;
360         }
361     };
362 };
363 } // namespace Hdc
364 #endif
365