1 /*
2 * Copyright (C) 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include <vector>
18
19 #include <gmock/gmock.h>
20
21 #include <application.h>
22 #include <nos/transport.h>
23
24 #include "crc16.h"
25
26 using ::testing::_;
27 using ::testing::Args;
28 using ::testing::DoAll;
29 using ::testing::Eq;
30 using ::testing::ElementsAreArray;
31 using ::testing::InSequence;
32 using ::testing::IsNull;
33 using ::testing::Return;
34 using ::testing::SetArrayArgument;
35 using ::testing::StrictMock;
36
37 namespace {
38
39 struct Device {
40 virtual int Read(uint32_t command, uint8_t* buf, uint32_t len) = 0;
41 virtual int Write(uint32_t command, const uint8_t* buf, uint32_t len) = 0;
42 virtual int WaitForInterrupt(int msecs) = 0;
43 virtual int Reset() = 0;
44 };
45
46 struct MockDevice : public Device {
47 MOCK_METHOD3(Read, int(uint32_t command, uint8_t* buf, uint32_t len));
48 MOCK_METHOD3(Write, int(uint32_t command, const uint8_t* buf, uint32_t len));
49 MOCK_METHOD1(WaitForInterrupt, int(int msecs));
50 MOCK_METHOD0(Reset, int());
51 };
52
53 // We want to closely examine the interactions with the device to make it a
54 // strict mock
55 using CtxType = StrictMock<MockDevice>;
56
57 // Forward calls onto the mock
read_datagram(void * ctx,uint32_t command,uint8_t * buf,uint32_t len)58 int read_datagram(void* ctx, uint32_t command, uint8_t* buf, uint32_t len) {
59 return reinterpret_cast<CtxType*>(ctx)->Read(command, buf, len);
60 }
write_datagram(void * ctx,uint32_t command,const uint8_t * buf,uint32_t len)61 int write_datagram(void* ctx, uint32_t command, const uint8_t* buf, uint32_t len) {
62 return reinterpret_cast<CtxType*>(ctx)->Write(command, buf, len);
63 }
wait_for_interrupt(void * ctx,int msecs)64 int wait_for_interrupt(void* ctx, int msecs) {
65 return reinterpret_cast<CtxType*>(ctx)->WaitForInterrupt(msecs);
66 }
reset(void * ctx)67 int reset(void* ctx) {
68 return reinterpret_cast<CtxType*>(ctx)->Reset();
69 }
close_device(void * ctx)70 void close_device(void* ctx) {
71 delete reinterpret_cast<CtxType*>(ctx);
72 }
73
74 // Implement the datagram API that calls a mock.
75 extern "C" {
nos_device_open(const char * device_name,struct nos_device * dev)76 int nos_device_open(const char* device_name, struct nos_device* dev) {
77 EXPECT_THAT(device_name, IsNull());
78 dev->ctx = new CtxType;
79 dev->ops.read = read_datagram;
80 dev->ops.write = write_datagram;
81 dev->ops.wait_for_interrupt = wait_for_interrupt;
82 dev->ops.reset = reset;
83 dev->ops.close = close_device;
84 return 0;
85 }
86 }
87
88 // Test fixture that sets up the mocked device.
89 struct TransportTest : public ::testing::Test {
SetUp__anon1f8664560111::TransportTest90 virtual void SetUp() override {
91 nos_device_open(nullptr, &dev_);
92 mock_dev_ = reinterpret_cast<CtxType*>(dev_.ctx);
93 }
TearDown__anon1f8664560111::TransportTest94 virtual void TearDown() override {
95 dev_.ops.close(dev_.ctx);
96 }
97
dev__anon1f8664560111::TransportTest98 nos_device* dev() { return &dev_; }
mock_dev__anon1f8664560111::TransportTest99 CtxType& mock_dev() { return *mock_dev_; }
100
101 private:
102 nos_device dev_;
103 CtxType* mock_dev_;
104 };
105
command_crc(uint32_t command,const uint8_t * args,uint16_t args_len,const transport_command_info * command_info)106 uint16_t command_crc(uint32_t command, const uint8_t* args, uint16_t args_len,
107 const transport_command_info* command_info) {
108 uint16_t crc = crc16(&args_len, sizeof(args_len));
109 crc = crc16_update(args, args_len, crc);
110 crc = crc16_update(&command, sizeof(command), crc);
111 crc = crc16_update(command_info, sizeof(*command_info), crc);
112 return htole16(crc);
113 }
114
115 } // namespace
116
117 /* Actions to return mock data */
118
119 #define READ_UNSET 0xdf
120
ACTION(ReadStatusV0_Idle)121 ACTION(ReadStatusV0_Idle) {
122 transport_status* status = (transport_status*)arg1;
123 memset(status, READ_UNSET, sizeof(*status));
124 status->status = APP_STATUS_IDLE;
125 status->reply_len = 0;
126 }
127
ACTION(ReadStatusV1_Idle)128 ACTION(ReadStatusV1_Idle) {
129 transport_status* status = (transport_status*)arg1;
130 memset(status, READ_UNSET, sizeof(*status));
131 status->status = APP_STATUS_IDLE;
132 status->reply_len = 0;
133 status->length = sizeof(transport_status);
134 status->version = TRANSPORT_V1;
135 status->flags = 0;
136 status->crc = STATUS_CRC_FOR_IDLE;
137 status->reply_crc = 0;
138 }
139
ACTION(ReadStatusV1_IdleWithBadCrc)140 ACTION(ReadStatusV1_IdleWithBadCrc) {
141 transport_status* status = (transport_status*)arg1;
142 memset(status, READ_UNSET, sizeof(*status));
143 status->status = APP_STATUS_IDLE;
144 status->reply_len = 0;
145 status->length = sizeof(transport_status);
146 status->version = TRANSPORT_V1;
147 status->flags = 0;
148 status->crc = STATUS_CRC_FOR_IDLE + 1; // <- wrong!
149 status->reply_crc = 0;
150 }
151
ACTION(ReadStatusV1_Working)152 ACTION(ReadStatusV1_Working) {
153 transport_status* status = (transport_status*)arg1;
154 memset(status, READ_UNSET, sizeof(*status));
155 status->status = APP_STATUS_IDLE;
156 status->reply_len = 0;
157 status->length = sizeof(transport_status);
158 status->version = TRANSPORT_V1;
159 status->flags = STATUS_FLAG_WORKING;
160 status->crc = STATUS_CRC_FOR_WORKING;
161 status->reply_crc = 0;
162 }
163
ACTION_P(ReadStatusV0_DoneWithData,reply_len)164 ACTION_P(ReadStatusV0_DoneWithData, reply_len) {
165 transport_status* status = (transport_status*)arg1;
166 memset(status, READ_UNSET, sizeof(*status));
167 status->status = APP_STATUS_DONE | APP_SUCCESS;
168 status->reply_len = reply_len;
169 }
170
ACTION_P2(ReadStatusV1_DoneWithData,reply,reply_len)171 ACTION_P2(ReadStatusV1_DoneWithData, reply, reply_len) {
172 transport_status* status = (transport_status*)arg1;
173 memset(status, READ_UNSET, sizeof(*status));
174 status->status = APP_STATUS_DONE | APP_SUCCESS;
175 status->reply_len = reply_len;
176 status->length = sizeof(transport_status);
177 status->version = TRANSPORT_V1;
178 status->flags = 0;
179 status->reply_crc = crc16(reply, reply_len);
180 status->crc = 0;
181 status->crc = crc16(status, status->length);
182 }
183
ACTION(ReadStatusV1_BadCrc)184 ACTION(ReadStatusV1_BadCrc) {
185 transport_status* status = (transport_status*)arg1;
186 memset(status, READ_UNSET, sizeof(*status));
187 status->status = APP_STATUS_DONE | APP_ERROR_CHECKSUM;
188 status->reply_len = 0;
189 status->length = sizeof(transport_status);
190 status->version = TRANSPORT_V1;
191 status->flags = 0;
192 status->crc = 0x92c0;
193 status->reply_crc = 0;
194 }
195
ACTION(ReadStatusV42_Working)196 ACTION(ReadStatusV42_Working) {
197 memset(arg1, 0xb3, STATUS_MAX_LENGTH);
198 transport_status* status = (transport_status*)arg1;
199 status->status = APP_STATUS_IDLE;
200 status->reply_len = 0;
201 status->length = STATUS_MAX_LENGTH;
202 status->version = 42;
203 status->flags = STATUS_FLAG_WORKING;
204 status->crc = 0xaec0;
205 status->reply_crc = 0;
206 }
207
ACTION_P3(ReadData,len,data,size)208 ACTION_P3(ReadData, len, data, size) {
209 memset(arg1, READ_UNSET, len);
210 memcpy(arg1, data, size);
211 }
212
213 /* Helper macros to expect datagram calls */
214
215 #define EXPECT_GET_STATUS_V0_IDLE(app_id) do { \
216 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
217 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
218 .WillOnce(DoAll(ReadStatusV0_Idle(), Return(0))); \
219 } while (0)
220
221 #define EXPECT_GET_STATUS_IDLE(app_id) do { \
222 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
223 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
224 .WillOnce(DoAll(ReadStatusV1_Idle(), Return(0))); \
225 } while (0)
226
227 #define EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id) do { \
228 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
229 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
230 .WillOnce(DoAll(ReadStatusV1_IdleWithBadCrc(), Return(0))); \
231 } while (0)
232
233 #define EXPECT_GET_STATUS_BAD_CRC(app_id) do { \
234 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
235 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
236 .WillOnce(DoAll(ReadStatusV1_BadCrc(), Return(0))); \
237 } while (0)
238
239 #define EXPECT_GET_STATUS_WORKING(app_id) do { \
240 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
241 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
242 .WillOnce(DoAll(ReadStatusV1_Working(), Return(0))); \
243 } while (0)
244
245 #define EXPECT_GET_STATUS_V0_DONE(app_id) do { \
246 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
247 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
248 .WillOnce(DoAll(ReadStatusV0_DoneWithData(0), Return(0))); \
249 } while (0)
250
251 #define EXPECT_GET_STATUS_V0_DONE_WITH_DATA(app_id, reply_len) do { \
252 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
253 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
254 .WillOnce(DoAll(ReadStatusV0_DoneWithData((reply_len)), Return(0))); \
255 } while (0)
256
257 #define EXPECT_GET_STATUS_DONE(app_id) do { \
258 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
259 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
260 .WillOnce(DoAll(ReadStatusV1_DoneWithData(nullptr, 0), Return(0))); \
261 } while (0)
262
263 #define EXPECT_GET_STATUS_DONE_WITH_DATA(app_id, reply, reply_len) do { \
264 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
265 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
266 .WillOnce(DoAll(ReadStatusV1_DoneWithData((reply), (reply_len)), Return(0))); \
267 } while (0)
268
269 #define EXPECT_GET_STATUS_DONE_BAD_CRC(app_id, reply, reply_len) do { \
270 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT; \
271 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH)) \
272 .WillOnce(DoAll(ReadStatusV1_DoneBadCrc((reply), (reply_len)), Return(0))); \
273 } while (0)
274
275 #define EXPECT_SEND_DATA(app_id, args, args_len) do { \
276 const uint32_t command = CMD_ID((app_id)) | CMD_IS_DATA | CMD_TRANSPORT | CMD_PARAM((args_len)); \
277 EXPECT_CALL(mock_dev(), Write(command, _, (args_len))) \
278 .With(Args<1,2>(ElementsAreArray((uint8_t*)(args), (args_len)))) \
279 .WillOnce(Return(0)); \
280 } while (0)
281
282 #define EXPECT_GO_COMMAND(app_id, param, args, args_len, reply_len) do { \
283 const uint32_t command = CMD_ID((app_id)) | CMD_PARAM((param)); \
284 transport_command_info command_info = {}; \
285 command_info.length = sizeof(command_info); \
286 command_info.version = htole16(TRANSPORT_V1); \
287 command_info.reply_len_hint = htole16((reply_len)); \
288 command_info.crc = command_crc(command, (args), (args_len), &command_info); \
289 EXPECT_CALL(mock_dev(), Write(command, _, command_info.length)) \
290 .With(Args<1,2>(ElementsAreArray((uint8_t*)&command_info, command_info.length))) \
291 .WillOnce(Return(0)); \
292 } while (0)
293
294 #define EXPECT_RECV_DATA(app_id, len, reply, reply_len) do { \
295 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_IS_DATA | CMD_TRANSPORT; \
296 EXPECT_CALL(mock_dev(), Read(command, _, (reply_len))) \
297 .WillOnce(DoAll(ReadData((len), (reply), (reply_len)), Return(0))); \
298 } while (0)
299
300 #define EXPECT_RECV_MORE_DATA(app_id, len, reply, reply_len) do { \
301 const uint32_t command = \
302 CMD_ID((app_id)) | CMD_IS_READ | CMD_IS_DATA | CMD_MORE_TO_COME | CMD_TRANSPORT; \
303 EXPECT_CALL(mock_dev(), Read(command, _, (reply_len))) \
304 .WillOnce(DoAll(ReadData((len), (reply), (reply_len)), Return(0))); \
305 } while (0)
306
307 #define EXPECT_CLEAR_STATUS(app_id) do { \
308 const uint32_t command = CMD_ID((app_id)) | CMD_TRANSPORT; \
309 EXPECT_CALL(mock_dev(), Write(command, _, 0)) \
310 .WillOnce(Return(0)); \
311 } while (0)
312
313 /* Protocol tests */
314
TEST_F(TransportTest,WorkingAppIsBusy)315 TEST_F(TransportTest, WorkingAppIsBusy) {
316 const uint8_t app_id = 213;
317 EXPECT_GET_STATUS_WORKING(app_id);
318
319 const uint16_t param = 2;
320 uint32_t reply_len = 0;
321 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, nullptr, &reply_len);
322 EXPECT_THAT(res, Eq(APP_ERROR_BUSY));
323 }
324
TEST_F(TransportTest,WorkingIsForwardCompatible)325 TEST_F(TransportTest, WorkingIsForwardCompatible) {
326 const uint8_t app_id = 25;
327 const uint32_t command = CMD_ID(app_id) | CMD_IS_READ | CMD_TRANSPORT;
328 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH))
329 .WillOnce(DoAll(ReadStatusV42_Working(), Return(0)));
330
331 const uint16_t param = 2;
332 uint32_t reply_len = 0;
333 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, nullptr, &reply_len);
334 EXPECT_THAT(res, Eq(APP_ERROR_BUSY));
335 }
336
TEST_F(TransportTest,SuccessIfStatusNotClear)337 TEST_F(TransportTest, SuccessIfStatusNotClear) {
338 const uint8_t app_id = 12;
339 const uint16_t param = 2;
340 const uint8_t args[] = {1, 2, 3};
341 const uint16_t args_len = 3;
342
343 InSequence please;
344 EXPECT_GET_STATUS_BAD_CRC(app_id);
345 // Try and reset
346 EXPECT_CLEAR_STATUS(app_id);
347 // Try again
348 EXPECT_GET_STATUS_IDLE(app_id);
349 EXPECT_SEND_DATA(app_id, args, args_len);
350 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
351 EXPECT_GET_STATUS_WORKING(app_id);
352 EXPECT_GET_STATUS_DONE(app_id);
353 EXPECT_CLEAR_STATUS(app_id);
354
355 uint32_t res = nos_call_application(dev(), app_id, param, args, args_len, nullptr, nullptr);
356 EXPECT_THAT(res, Eq(APP_SUCCESS));
357 }
358
TEST_F(TransportTest,StatusCrcError)359 TEST_F(TransportTest, StatusCrcError) {
360 const uint8_t app_id = 53;
361 const uint16_t param = 192;
362
363 InSequence please;
364 // Try 5 times
365 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
366 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
367 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
368 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
369 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
370
371 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, nullptr, nullptr);
372 EXPECT_THAT(res, Eq(APP_ERROR_IO));
373 }
374
TEST_F(TransportTest,FailToClearStatus)375 TEST_F(TransportTest, FailToClearStatus) {
376 const uint8_t app_id = 12;
377 const uint16_t param = 2;
378 const uint8_t args[] = {1, 2, 3};
379 const uint16_t args_len = 3;
380
381 InSequence please;
382 EXPECT_GET_STATUS_BAD_CRC(app_id);
383 // Try and reset
384 EXPECT_CLEAR_STATUS(app_id);
385 // No luck
386 EXPECT_GET_STATUS_BAD_CRC(app_id);
387
388 uint32_t res = nos_call_application(dev(), app_id, param, args, args_len, nullptr, nullptr);
389 EXPECT_THAT(res, Eq(APP_ERROR_IO));
390 }
391
TEST_F(TransportTest,FailToClearStatusAfterStatusCrcError)392 TEST_F(TransportTest, FailToClearStatusAfterStatusCrcError) {
393 const uint8_t app_id = 53;
394 const uint16_t param = 192;
395
396 InSequence please;
397 // Try 5 times
398 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
399 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
400 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
401 EXPECT_GET_STATUS_IDLE_WITH_BAD_CRC(app_id);
402 EXPECT_GET_STATUS_BAD_CRC(app_id);
403 // Try and reset
404 EXPECT_CLEAR_STATUS(app_id);
405 // No luck
406 EXPECT_GET_STATUS_BAD_CRC(app_id);
407
408 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, nullptr, nullptr);
409 EXPECT_THAT(res, Eq(APP_ERROR_IO));
410 }
411
TEST_F(TransportTest,RequestCrcError)412 TEST_F(TransportTest, RequestCrcError) {
413 const uint8_t app_id = 58;
414 const uint16_t param = 93;
415 const uint8_t args[] = {4, 24, 183, 255, 219};
416 const uint16_t args_len = 5;
417
418 InSequence please;
419 // Should try 5 times
420 EXPECT_GET_STATUS_IDLE(app_id);
421 EXPECT_SEND_DATA(app_id, args, args_len);
422 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
423 EXPECT_GET_STATUS_BAD_CRC(app_id);
424 // 4 more
425 EXPECT_GET_STATUS_IDLE(app_id);
426 EXPECT_SEND_DATA(app_id, args, args_len);
427 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
428 EXPECT_GET_STATUS_BAD_CRC(app_id);
429 // 3 more
430 EXPECT_GET_STATUS_IDLE(app_id);
431 EXPECT_SEND_DATA(app_id, args, args_len);
432 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
433 EXPECT_GET_STATUS_BAD_CRC(app_id);
434 // 2 more
435 EXPECT_GET_STATUS_IDLE(app_id);
436 EXPECT_SEND_DATA(app_id, args, args_len);
437 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
438 EXPECT_GET_STATUS_BAD_CRC(app_id);
439 // last one
440 EXPECT_GET_STATUS_IDLE(app_id);
441 EXPECT_SEND_DATA(app_id, args, args_len);
442 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
443 EXPECT_GET_STATUS_BAD_CRC(app_id);
444 // Clean up
445 EXPECT_CLEAR_STATUS(app_id);
446
447 uint32_t res = nos_call_application(dev(), app_id, param, args, args_len, nullptr, nullptr);
448 EXPECT_THAT(res, Eq(APP_ERROR_IO));
449 }
450
TEST_F(TransportTest,SuccessAfterRequestCrcError)451 TEST_F(TransportTest, SuccessAfterRequestCrcError) {
452 const uint8_t app_id = 255;
453 const uint16_t param = 163;
454 const uint8_t args[] = {42, 89, 125, 0, 83, 92, 80};
455 const uint16_t args_len = 7;
456
457 InSequence please;
458 // First request is CRC error
459 EXPECT_GET_STATUS_IDLE(app_id);
460 EXPECT_SEND_DATA(app_id, args, args_len);
461 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
462 EXPECT_GET_STATUS_BAD_CRC(app_id);
463 // The retry succeeds
464 EXPECT_GET_STATUS_IDLE(app_id);
465 EXPECT_SEND_DATA(app_id, args, args_len);
466 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
467 EXPECT_GET_STATUS_WORKING(app_id);
468 EXPECT_GET_STATUS_DONE(app_id);
469 EXPECT_CLEAR_STATUS(app_id);
470
471 uint32_t res = nos_call_application(dev(), app_id, param, args, args_len, nullptr, nullptr);
472 EXPECT_THAT(res, Eq(APP_SUCCESS));
473 }
474
TEST_F(TransportTest,SuccessWithoutReply)475 TEST_F(TransportTest, SuccessWithoutReply) {
476 const uint8_t app_id = 12;
477 const uint16_t param = 2;
478 const uint8_t args[] = {1, 2, 3};
479 const uint16_t args_len = 3;
480
481 InSequence please;
482 EXPECT_GET_STATUS_IDLE(app_id);
483 EXPECT_SEND_DATA(app_id, args, args_len);
484 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
485 EXPECT_GET_STATUS_WORKING(app_id);
486 EXPECT_GET_STATUS_DONE(app_id);
487 EXPECT_CLEAR_STATUS(app_id);
488
489 uint32_t res = nos_call_application(dev(), app_id, param, args, args_len, nullptr, nullptr);
490 EXPECT_THAT(res, Eq(APP_SUCCESS));
491 }
492
TEST_F(TransportTest,DetectAppAbort)493 TEST_F(TransportTest, DetectAppAbort) {
494 const uint8_t app_id = 25;
495 const uint16_t param = 252;
496 const uint8_t args[] = {17, 27, 43, 193};
497 const uint16_t args_len = 4;
498
499 InSequence please;
500 EXPECT_GET_STATUS_IDLE(app_id);
501 EXPECT_SEND_DATA(app_id, args, args_len);
502 EXPECT_GO_COMMAND(app_id, param, args, args_len, 0);
503 EXPECT_GET_STATUS_WORKING(app_id);
504 EXPECT_GET_STATUS_WORKING(app_id);
505 EXPECT_GET_STATUS_WORKING(app_id);
506 // It just stopped working
507 EXPECT_GET_STATUS_IDLE(app_id);
508 // It's probably already clear but just making sure
509 EXPECT_CLEAR_STATUS(app_id);
510
511 uint32_t res = nos_call_application(dev(), app_id, param, args, args_len, nullptr, nullptr);
512 EXPECT_THAT(res, Eq(APP_ERROR_INTERNAL));
513 }
514
TEST_F(TransportTest,SuccessWithReply)515 TEST_F(TransportTest, SuccessWithReply) {
516 const uint8_t app_id = 165;
517 const uint16_t param = 16;
518 const uint8_t data[] = {5, 6, 7, 8};
519 uint8_t reply[4];
520 uint32_t reply_len = 4;
521
522 InSequence please;
523 EXPECT_GET_STATUS_IDLE(app_id);
524 EXPECT_SEND_DATA(app_id, nullptr, 0);
525 EXPECT_GO_COMMAND(app_id, param, nullptr, 0, reply_len);
526 EXPECT_GET_STATUS_DONE_WITH_DATA(app_id, data, sizeof(data));
527 EXPECT_RECV_DATA(app_id, reply_len, data, sizeof(data));
528 EXPECT_CLEAR_STATUS(app_id);
529
530 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, reply, &reply_len);
531 EXPECT_THAT(res, Eq(APP_SUCCESS));
532 EXPECT_THAT(reply_len, Eq(4));
533 EXPECT_THAT(reply, ElementsAreArray(data, sizeof(data)));
534 }
535
TEST_F(TransportTest,SuccessWithReplyInMultipleDatagrams)536 TEST_F(TransportTest, SuccessWithReplyInMultipleDatagrams) {
537 const uint8_t app_id = 165;
538 const uint16_t param = 16;
539 std::vector<uint8_t> data(MAX_DEVICE_TRANSFER + 24, 0xea);
540 std::vector<uint8_t> reply(data.size());
541 uint32_t reply_len = reply.size();
542
543 InSequence please;
544 EXPECT_GET_STATUS_IDLE(app_id);
545 EXPECT_SEND_DATA(app_id, nullptr, 0);
546 EXPECT_GO_COMMAND(app_id, param, nullptr, 0, reply_len);
547 EXPECT_GET_STATUS_DONE_WITH_DATA(app_id, data.data(), data.size());
548 EXPECT_RECV_DATA(app_id, reply_len, data.data(), MAX_DEVICE_TRANSFER);
549 EXPECT_RECV_MORE_DATA(app_id, 24, data.data() + MAX_DEVICE_TRANSFER, 24);
550 EXPECT_CLEAR_STATUS(app_id);
551
552 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, reply.data(), &reply_len);
553 EXPECT_THAT(res, Eq(APP_SUCCESS));
554 EXPECT_THAT(reply_len, Eq(MAX_DEVICE_TRANSFER + 24));
555 EXPECT_THAT(reply, ElementsAreArray(data));
556 }
557
TEST_F(TransportTest,ReplyCrcError)558 TEST_F(TransportTest, ReplyCrcError) {
559 const uint8_t app_id = 5;
560 const uint16_t param = 0;
561 const uint8_t data[] = {1, 1, 2, 3, 5, 7};
562 const uint8_t wrong_data[] = {3, 1, 2, 3, 5, 7};
563 uint8_t reply[6];
564 uint32_t reply_len = 6;
565
566 InSequence please;
567 EXPECT_GET_STATUS_IDLE(app_id);
568 EXPECT_SEND_DATA(app_id, nullptr, 0);
569 EXPECT_GO_COMMAND(app_id, param, nullptr, 0, reply_len);
570 EXPECT_GET_STATUS_DONE_WITH_DATA(app_id, data, sizeof(data));
571 // Try 5 times to read data
572 EXPECT_RECV_DATA(app_id, reply_len, wrong_data, sizeof(wrong_data));
573 EXPECT_RECV_DATA(app_id, reply_len, wrong_data, sizeof(wrong_data));
574 EXPECT_RECV_DATA(app_id, reply_len, wrong_data, sizeof(wrong_data));
575 EXPECT_RECV_DATA(app_id, reply_len, wrong_data, sizeof(wrong_data));
576 EXPECT_RECV_DATA(app_id, reply_len, wrong_data, sizeof(wrong_data));
577
578 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, reply, &reply_len);
579 EXPECT_THAT(res, Eq(APP_ERROR_IO));
580 }
581
TEST_F(TransportTest,SuccessAfterReplyCrcError)582 TEST_F(TransportTest, SuccessAfterReplyCrcError) {
583 const uint8_t app_id = 5;
584 const uint16_t param = 0;
585 const uint8_t data[] = {2, 4, 9, 16};
586 const uint8_t wrong_data[] = {2, 4, 9, 48};
587 uint8_t reply[4];
588 uint32_t reply_len = 4;
589
590 InSequence please;
591 EXPECT_GET_STATUS_IDLE(app_id);
592 EXPECT_SEND_DATA(app_id, nullptr, 0);
593 EXPECT_GO_COMMAND(app_id, param, nullptr, 0, reply_len);
594 EXPECT_GET_STATUS_DONE_WITH_DATA(app_id, data, sizeof(data));
595 // Retry due to crc error
596 EXPECT_RECV_DATA(app_id, reply_len, wrong_data, sizeof(wrong_data));
597 EXPECT_RECV_DATA(app_id, reply_len, wrong_data, sizeof(wrong_data));
598 EXPECT_RECV_DATA(app_id, reply_len, data, sizeof(data));
599 EXPECT_CLEAR_STATUS(app_id);
600
601 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, reply, &reply_len);
602 EXPECT_THAT(res, Eq(APP_SUCCESS));
603 EXPECT_THAT(reply_len, Eq(4));
604 EXPECT_THAT(reply, ElementsAreArray(data, sizeof(data)));
605 }
606
TEST_F(TransportTest,V0SuccessWithoutReply)607 TEST_F(TransportTest, V0SuccessWithoutReply) {
608 const uint8_t app_id = 6;
609 const uint16_t param = 92;
610
611 InSequence please;
612 EXPECT_GET_STATUS_V0_IDLE(app_id);
613 EXPECT_SEND_DATA(app_id, nullptr, 0);
614 EXPECT_GO_COMMAND(app_id, param, nullptr, 0, 0);
615 EXPECT_GET_STATUS_V0_DONE(app_id);
616 EXPECT_CLEAR_STATUS(app_id);
617
618 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, nullptr, nullptr);
619 EXPECT_THAT(res, Eq(APP_SUCCESS));
620 }
621
TEST_F(TransportTest,V0SuccessWithReply)622 TEST_F(TransportTest, V0SuccessWithReply) {
623 const uint8_t app_id = 0;
624 const uint16_t param = 18;
625 const uint8_t data[] = {15, 20, 25, 30, 35, 40};
626 uint8_t reply[6];
627 uint32_t reply_len = 6;
628
629 InSequence please;
630 EXPECT_GET_STATUS_V0_IDLE(app_id);
631 EXPECT_SEND_DATA(app_id, nullptr, 0);
632 EXPECT_GO_COMMAND(app_id, param, nullptr, 0, reply_len);
633 EXPECT_GET_STATUS_V0_DONE_WITH_DATA(app_id, sizeof(data));
634 EXPECT_RECV_DATA(app_id, reply_len, data, sizeof(data));
635 EXPECT_CLEAR_STATUS(app_id);
636
637 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, reply, &reply_len);
638 EXPECT_THAT(res, Eq(APP_SUCCESS));
639 EXPECT_THAT(reply_len, Eq(6));
640 EXPECT_THAT(reply, ElementsAreArray(data, sizeof(data)));
641 }
642
TEST_F(TransportTest,ErrorIfArgsLenButNotArgs)643 TEST_F(TransportTest, ErrorIfArgsLenButNotArgs) {
644 uint8_t reply[] = {1, 2, 3};
645 uint32_t reply_len = 0;
646 uint32_t status = nos_call_application(dev(), 1, 2, nullptr, 5, reply, &reply_len);
647 EXPECT_THAT(status, Eq(APP_ERROR_IO));
648 }
649
main(int argc,char ** argv)650 int main(int argc, char** argv) {
651 ::testing::InitGoogleTest(&argc, argv);
652 return RUN_ALL_TESTS();
653 }
654
655 #ifdef TEST_TIMEOUT
TEST_F(TransportTest,Timeout)656 TEST_F(TransportTest, Timeout) {
657 const uint8_t app_id = 49;
658 const uint16_t param = 64;
659
660 InSequence please;
661 EXPECT_GET_STATUS_IDLE(app_id);
662 EXPECT_SEND_DATA(app_id, nullptr, 0);
663 EXPECT_GO_COMMAND(app_id, param, nullptr, 0, 0);
664
665 // Keep saying we're working on it
666 const uint32_t command = CMD_ID((app_id)) | CMD_IS_READ | CMD_TRANSPORT;
667 EXPECT_CALL(mock_dev(), Read(command, _, STATUS_MAX_LENGTH))
668 .WillRepeatedly(DoAll(ReadStatusV1_Working(), Return(0)));
669
670 // We'll still try and clean up
671 EXPECT_CLEAR_STATUS(app_id);
672
673 uint32_t res = nos_call_application(dev(), app_id, param, nullptr, 0, nullptr, nullptr);
674 EXPECT_THAT(res, Eq(APP_ERROR_TIMEOUT));
675 }
676 #endif
677