blob: baf378c6a3bd15dd212b257488ec38ed67cec7b2 [file] [log] [blame]
zongqiang.zhang0c6a0882019-08-07 14:48:21 +08001#include "sp_communicate.h"
2#include "sp_util.h"
3#include "sp_flash.h"
4#include "sp_constant.h"
5#include "sp_data.h"
6#include "sp_msgpack.h"
7#include "sp_display.h"
guangcheng.qin55f3fa62019-08-07 16:14:46 +08008#include "sp_upgrade.h"
zongqiang.zhang0c6a0882019-08-07 14:48:21 +08009#include "../sys_hw/drv_usart.h"
10
11static void sp_usart_send(sp_pos_t* pos, sp_protocol_request_t* req)
12{
13 uint8 sendBuff[264];
14 MEMCLEAR(sendBuff, sizeof(sendBuff));
15
16 pos->last_comm_status.command = req->excmd;
17 pos->last_comm_status.sendtime = sp_get_ticker();
18
19 sp_protocol_crc((uint8*)req +2, req->datalen, (uint8*)req +2 +req->datalen);
20 req->datalen += 2;
21 MEMCPY(sendBuff, req, req->datalen+2);
22 usart_send(sendBuff, req->datalen+2);
23}
24
25static uint8 sp_usart_recv(sp_pos_t* pos, sp_protocol_response_t* resp, int32 timeout_ms)
26{
27 uint32 tick = 0;
28 tick = sp_get_ticker();
29 while(1)
30 {
31 MEMCLEAR(resp, sizeof(sp_protocol_response_t));
32 usart_read((u8*)resp, sizeof(sp_protocol_response_t));
33 if(pos->last_comm_status.command == resp->excmd)
34 {
35 MEMCLEAR(&(pos->last_comm_status), sizeof(sp_comm_status_t));
36 return resp->retcode;
37 }
38 if((sp_get_ticker() - tick) >= timeout_ms)
39 {
40 return 1;
41 }
42 }
43}
44uint8 sp_comm_call(sp_pos_t* pos, sp_protocol_request_t* req,
45 sp_protocol_response_t* resp, int32 timeout_ms)
46{
47 sp_usart_send(pos, req);
48 return sp_usart_recv(pos, resp, timeout_ms);
49}
50
51void sp_protocol_req_init(sp_protocol_request_t* req, uint8 command)
52{
53 MEMCLEAR(req,sizeof(sp_protocol_request_t));
54 req->command = PROTOCOL_COMMAND_V2;
55 req->excmd = command;
56 req->flag = PROTOCOL_FLAG_PACK(req->flag);
57 req->flag = PROTOCOL_WITHOUT_MAC(req->flag);
58 req->datalen = 3;
59}
60
61static uint16 sp_confirm_card_authentication(sp_pos_t* pos, sp_card_t* card)
62{
63 uint8 size;
64 uint16 ret;
65 int32 timeout_ms = COMM_WAIT_TIME;
66 cw_unpack_context unpack;
67 unpack_field_t field;
68 sp_protocol_response_t resp;
69
70 disp_hint_info(pos,"ÕýÔÚÉí·ÝÈÏÖ¤ÖÐ",DELAY_TIME2s);
71 ret = sp_usart_recv(pos, &resp, timeout_ms);
72 if(ret)
73 {
74 ret = RC_CARD_AUTHENTICATION;
75 return ret;
76 }
77
78 sp_unpack_init(&unpack,resp.data,resp.datalen);
79 size = sp_unpack_map_size(&unpack);
80 while(size-- > 0)
81 {
82 sp_unpack_value(&unpack,&field);
83 if(IS_KEY(PK_INT_WATERLIMIT,field.key))
84 {
85 card->waterlimit = field.val.intval;
86 }
87 else if(IS_KEY(PK_BIN_BILLNO,field.key))
88 {
89 MEMCPY(card->billno, field.val.binval, field.strlen);
90 }
91 else if(IS_KEY(PK_INT_FEEAMOUNT,field.key))
92 {
93 card->feepara.fee_amt = field.val.intval;
94 }
95 else if(IS_KEY(PK_INT_FEEUNIT, field.key))
96 {
97 card->feepara.fee_unit = field.val.intval;
98 }
99 else if(IS_KEY(PK_INT_FEESTART, field.key))
100 {
101 card->feepara.fee_start = field.val.intval;
102 }
103 }
104 disp_hint_info(pos,"Éí·ÝÈÏÖ¤³É¹¦",DELAY_TIME1s);
105 return 0;
106}
107
108//ºǫ́¶Ô¿¨µÄÉí·ÝÈÏÖ¤
109uint16 sp_card_authentication(sp_pos_t* pos, sp_card_t* card)
110{
111 uint8 ctime[6];
112 sp_protocol_request_t req;
113 cw_pack_context pack;
114 MEMCLEAR(&req, sizeof(req));
115 MEMCLEAR(&pack, sizeof(req));
116 MEMCLEAR(ctime, sizeof ctime);
117
118 sp_get_bcdtime(ctime);
119 sp_protocol_req_init(&req, SP_CMD_CARD_AUTHENTICATION);
120 sp_pack_init(&pack, req.data, sizeof(req.data));
121 cw_pack_map_size(&pack,4);
122
123 sp_pack_put_bin(&pack, PK_BIN_CARDPHYID, card->cardphyid, 4);
124 sp_pack_put_bin(&pack, PK_BIN_CITIZEN_CARDNO, card->citizen_cardno, 12);
125 sp_pack_put_bin(&pack, PK_BIN_DEVPHYID, pos->devphyid, 4);
126 sp_pack_put_bin(&pack, PK_BIN_DEVTIME, ctime, 6);
127
128 req.datalen += sp_pack_length(&pack);
129 sp_usart_send(pos, &req);
130 return sp_confirm_card_authentication(pos, card);
131}
132
133//É豸ǩµ½
134uint16 sp_async_equipment_login(sp_pos_t* pos)
135{
136 sp_protocol_request_t req;
137 cw_pack_context pack;
138 uint8 ctime[6];
139 MEMCLEAR(ctime, sizeof(ctime));
140 pos->heartbeat.heart_status = HEART_SEND;
141
142 sp_get_bcdtime(ctime);
143 sp_protocol_req_init(&req, SP_CMD_LOGIN);
144 sp_pack_init(&pack, req.data, sizeof(req.data));
145 cw_pack_map_size(&pack, 5);
146
147 sp_pack_put_bin(&pack, PK_BIN_DEVPHYID, pos->devphyid, 4);
148 sp_pack_put_bin(&pack, PK_BIN_DEVTIME, ctime, 6);
149 sp_pack_put_str(&pack, PK_STR_DEVTYPE, DEV_TYPE);
150 sp_pack_put_str(&pack, PK_STR_VERSION, PRO_VERSION);
151 sp_pack_put_bin(&pack, PK_BIN_DEVICEKEY, (pos->sysconf.deviceKey), 8);
152
153 req.datalen += sp_pack_length(&pack);
154 sp_usart_send(pos, &req);
155 return 0;
156}
157
158//ÐÄÌøÈ·ÈÏ£¬¼ì²âÍøÂçÊÇ·ñÕý³£
159uint16 sp_async_heartbeat(sp_pos_t* pos)
160{
161 uint8 ctime[6];
162 uint8 unconfirm_transnum = 0;
163 sp_protocol_request_t req;
164 cw_pack_context pack;
165
166 pos->heartbeat.heart_status = HEART_SEND;
167 if(pos->unconfirm_transdtl.transaddr <= pos->last_transdtl.transaddr)
168 {
169 unconfirm_transnum = ((pos->last_transdtl.transaddr - pos->unconfirm_transdtl.transaddr) /
170 sizeof(sp_transdtl_t)) + 1;
171 }
172 else
173 {
174 unconfirm_transnum = 0;
175 }
176 sp_protocol_req_init(&req, SP_CMD_HEARTBEAT);
177 sp_pack_init(&pack, req.data, sizeof(req.data));
178 cw_pack_map_size(&pack, 6);
179
180 sp_get_bcdtime(ctime);
181 sp_pack_put_bin(&pack, PK_BIN_DEVPHYID, pos->devphyid, 4);
182 sp_pack_put_bin(&pack, PK_BIN_DEVTIME, ctime, 6);
183 sp_pack_put_str(&pack, PK_STR_DEVTYPE, DEV_TYPE);
184 sp_pack_put_str(&pack, PK_STR_VERSION, PRO_VERSION);
185 sp_pack_put_int(&pack, PK_INT_UNTRANSCONST, unconfirm_transnum);
186 sp_pack_put_int(&pack, PK_INT_WORKMODE, pos->sysconf.work_mode);
187
188 req.datalen += sp_pack_length(&pack);
189 sp_usart_send(pos, &req);
190 return 0;
191}
192
193static uint16 sp_confirm_qrcode_init(sp_pos_t* pos, sp_card_t* card)
194{
195 uint8 size;
196 uint16 ret;
197 int32 timeout_ms = COMM_WAIT_TIME;
198 cw_unpack_context unpack;
199 unpack_field_t field;
200 sp_protocol_response_t resp;
201
202 ret = sp_usart_recv(pos, &resp, timeout_ms);
203 if(ret)
204 {
205 ret = RC_QRCODE_FAILURE;
206 return ret;
207 }
208
209 sp_unpack_init(&unpack,resp.data,resp.datalen);
210 size = sp_unpack_map_size(&unpack);
211 while(size-- > 0)
212 {
213 sp_unpack_value(&unpack,&field);
214 if(IS_KEY(PK_STR_SHORT_URL, field.key))
215 {
216 MEMCPY(card->qrcode.qrcode_url, field.val.strval, field.strlen);
217 }
218 else if(IS_KEY(PK_BIN_BILLNO, field.key))
219 {
220 MEMCPY(card->billno, field.val.binval, field.strlen);
221 }
222 else if(IS_KEY(PK_INT_VAILDTIME, field.key))
223 {
224 card->qrcode.validtime = field.val.intval;
225 }
226 }
227 card->qrcode.starttime = sp_get_ticker();
228 return 0;
229}
230
231//¶þάÂë»ñÈ¡
232uint16 sp_qrcode_init(sp_pos_t* pos, sp_card_t* card)
233{
234 uint8 ctime[6];
235 sp_protocol_request_t req;
236 cw_pack_context pack;
237
238 sp_get_bcdtime(ctime);
239 sp_protocol_req_init(&req, SP_CMD_SHORTURL);
240 sp_pack_init(&pack, req.data, sizeof(req.data));
241 cw_pack_map_size(&pack, 2);
242
243 sp_pack_put_bin(&pack, PK_BIN_DEVPHYID, pos->devphyid, 4);
244 sp_pack_put_bin(&pack, PK_BIN_DEVTIME, ctime, 6);
245
246 req.datalen += sp_pack_length(&pack);
247 sp_usart_send(pos, &req);
248 return sp_confirm_qrcode_init(pos, card);
249}
250
251//¶þάÂëÈ·ÈÏ£¬»ñȡ֧¸¶×´Ì¬
252static uint16 sp_confirm_qrcode_query(sp_pos_t* pos, sp_card_t* card)
253{
254 uint8 size;
255 uint16 ret;
256 int32 timeout_ms = COMM_WAIT_TIME;
257 cw_unpack_context unpack;
258 unpack_field_t field;
259 sp_protocol_response_t resp;
260
261 ret = sp_usart_recv(pos, &resp, timeout_ms);
262 if(ret)
263 {
264 ret = RC_QRCODE_QUERY_FAIL;
265 return ret;
266 }
267
268 sp_unpack_init(&unpack,resp.data,resp.datalen);
269 size = sp_unpack_map_size(&unpack);
270 while(size-- > 0)
271 {
272 sp_unpack_value(&unpack,&field);
273 if(IS_KEY(PK_BIN_BILLNO, field.key))
274 {
275 MEMCPY(card->billno, field.val.binval, field.strlen);
276 }
277 else if(IS_KEY(PK_INT_AUTHSTATUS, field.key))
278 {
279 card->qrcode.authstatus = field.val.intval;
280 }
281 else if(IS_KEY(PK_INT_PAYSTATUS, field.key))
282 {
283 card->qrcode.paystatus = field.val.intval;
284 }
285 else if(IS_KEY(PK_INT_PAYAMT, field.key))
286 {
287 card->qrcode.paidAmount = field.val.intval;
288 }
289 else if(IS_KEY(PK_INT_FEEAMOUNT,field.key))
290 {
291 card->feepara.fee_amt = field.val.intval;
292 }
293 else if(IS_KEY(PK_INT_FEEUNIT, field.key))
294 {
295 card->feepara.fee_unit = field.val.intval;
296 }
297 }
298 return resp.retcode;
299}
300
301uint16 sp_qrcode_query(sp_pos_t* pos, sp_card_t* card)
302{
303 uint8 ctime[6];
304 sp_protocol_request_t req;
305 cw_pack_context pack;
306
307 sp_get_bcdtime(ctime);
308 sp_protocol_req_init(&req, SP_CMD_QRCODE_PAY_QUERY);
309 sp_pack_init(&pack, req.data, sizeof(req.data));
310 cw_pack_map_size(&pack, 2);
311
312 sp_pack_put_bin(&pack, PK_BIN_DEVPHYID, pos->devphyid, 4);
313 sp_pack_put_bin(&pack, PK_BIN_BILLNO, card->billno, sizeof(card->billno));
314
315 req.datalen += sp_pack_length(&pack);
316 sp_usart_send(pos, &req);
317 return sp_confirm_qrcode_query(pos, card);
318}
319
320//Á÷Ë®ÉÏ´«
321static uint8 transdtl_account_bilLno[10];
322static uint16 sp_async_upload_transdtl(sp_pos_t* pos, sp_transdtl_t* transdtl)
323{
324 uint8 crc[2];
325 cw_pack_context pack;
326 sp_protocol_request_t req;
327
328 MEMCPY(transdtl_account_bilLno, transdtl->billno, sizeof(transdtl->billno));
329 sp_protocol_req_init(&req, SP_CMD_TRANSDTL_ACCOUNT);
330 sp_pack_init(&pack,req.data,sizeof(req.data));
331 cw_pack_map_size(&pack,10);
332
333 sp_pack_put_bin(&pack, PK_BIN_DEVPHYID, transdtl->devphyid,4);
334 sp_pack_put_bin(&pack, PK_BIN_TRANSDATE, transdtl->transdate,3);
335 sp_pack_put_bin(&pack, PK_BIN_TRANSTIME, transdtl->transtime,3);
336 sp_pack_put_bin(&pack, PK_BIN_BILLNO, transdtl->billno, sizeof(transdtl->billno));
337 sp_pack_put_bin(&pack, PK_BIN_CARDPHYID, transdtl->cardphyid,
338 sizeof(transdtl->cardphyid));
339 sp_pack_put_int(&pack, PK_INT_TRANSWAY, transdtl->transway);
340 sp_pack_put_int(&pack, PK_INT_AMOUNT, transdtl->amount);
341 sp_pack_put_int(&pack, PK_INT_PAYAMT, transdtl->paidAmount);
342 sp_pack_put_int(&pack, PK_INT_FLOWSENSORS, transdtl->flowsensors);
343 sp_pack_put_int(&pack, PK_INT_FLAG, transdtl->transtatus);
344
345 sp_protocol_crc((uint8*)transdtl, sizeof(sp_transdtl_t)-2,crc);
346 req.datalen += sp_pack_length(&pack);
347 sp_usart_send(pos, &req);
348 return 0;
349}
350
351//È·ÈÏÁ÷ˮ״̬
352static uint8 sp_confirm_transdtl_account(sp_protocol_response_t* resp, sp_pos_t* pos)
353{
354 uint8 size;
355 cw_unpack_context unpack;
356 unpack_field_t field;
357
358 if(!resp->retcode)
359 {
360 sp_unpack_init(&unpack, resp->data, resp->datalen);
361 size = sp_unpack_map_size(&unpack);
362 while(size-- > 0)
363 {
364 sp_unpack_value(&unpack,&field);
365 if(IS_KEY(PK_BIN_BILLNO,field.key))
366 {
367 if(MEMCMP(field.val.binval, transdtl_account_bilLno,
368 sizeof(transdtl_account_bilLno)) == 0)
369 {
370 pos->unconfirm_transdtl.transaddr+= sizeof(sp_transdtl_t);
371 if(ADDR_TRANSDTL_END <= pos->unconfirm_transdtl.transaddr)
372 {
373 pos->unconfirm_transdtl.transaddr= ADDR_TRANSDTL_BEGIN;
374 }
375 return sp_write_unconfirm_record(pos);
376 }
377 }
378 }
379 }
380 return resp->retcode;
381
382}
383
384static uint8 sp_confirm_heartbeat(sp_protocol_response_t* resp, sp_pos_t* pos)
385{
386 uint8 size;
387 uint8 ctime[6];
388 uint8 systime[7];
389 uint8 login_flag = 0;
390 cw_unpack_context unpack;
391 unpack_field_t field;
392
393 if(resp->retcode)
394 {
395 return resp->retcode;
396 }
397 pos->heartbeat.heart_status = HEART_RECV;
398
399 sp_unpack_init(&unpack,resp->data,resp->datalen);
400 size = sp_unpack_map_size(&unpack);
401 while(size-- > 0)
402 {
403 sp_unpack_value(&unpack,&field);
404 if(IS_KEY(PK_BIN_SYSTIME, field.key))
405 {
406 if(field.strlen == 7)
407 {
408 MEMCPY(systime,field.val.binval,7);
409 sp_get_bcdtime(ctime);
410 if(MEMCMP(ctime,systime +1,5) != 0)
411 {
412 if(!sp_check_time_valid(systime +1))
413 {
414 sp_set_bcdtime(systime +1);
415 }
416 }
417 }
418 }
419 else if(IS_KEY(PK_STR_STATUS, field.key))
420 {
421 if(MEMCMP(field.val.strval, "normal", field.strlen) == 0)
422 {
423 login_flag = 1;
424 }
425 else if(MEMCMP(field.val.strval, "logout", field.strlen) == 0)
426 {
427 login_flag = 0;
428 }
429 else if(MEMCMP(field.val.strval, "closed", field.strlen) == 0)
430 {
431 login_flag = 2;
432 }
433 }
guangcheng.qin55f3fa62019-08-07 16:14:46 +0800434 else if(IS_KEY(PK_STR_UPGRADE, field.key))
435 {
436 if(MEMCMP("true",field.val.strval,4) == 0)
437 {
438 sp_manual_upgrade(pos);
439 }
440 }
zongqiang.zhang0c6a0882019-08-07 14:48:21 +0800441 }
442 sp_save_heartbeat_info(pos, login_flag);
443 return resp->retcode;
444}
445
446uint8 sp_confirm_login(sp_protocol_response_t* resp, sp_pos_t* pos)
447{
448 uint8 size;
449 uint8 ctime[6];
450 uint8 systime[7];
451 uint8 login_flag = 0;
452 uint8 unit = 0;
453 uint8 offline_maxhour = 0;
454 cw_unpack_context unpack;
455 unpack_field_t field;
456
457 MEMCLEAR(ctime, sizeof(ctime));
458 MEMCLEAR(systime, sizeof(systime));
459 if(resp->retcode)
460 {
461 return resp->retcode;
462 }
463 pos->heartbeat.heart_status = HEART_RECV;
464
465 sp_unpack_init(&unpack,resp->data,resp->datalen);
466 size = sp_unpack_map_size(&unpack);
467 while(size-- > 0)
468 {
469 sp_unpack_value(&unpack,&field);
470 if(IS_KEY(PK_INT_OFFLINEMAXHOUR, field.key))
471 {
472 offline_maxhour = field.val.intval;
473 }
474 else if(IS_KEY(PK_INT_PULSEINHML, field.key))
475 {
476 unit = field.val.intval;
477 }
478 else if(IS_KEY(PK_BIN_SYSTIME, field.key))
479 {
480 if(field.strlen == 7)
481 {
482 MEMCPY(systime,field.val.binval,7);
483 sp_get_bcdtime(ctime);
484 if(MEMCMP(ctime,systime +1,5) != 0)
485 {
486 if(!sp_check_time_valid(systime +1))
487 {
488 sp_set_bcdtime(systime +1);
489 }
490 }
491 }
492 }
493 }
494
495 login_flag = 1;
496 sp_save_login_info(pos, login_flag, unit, offline_maxhour);
497 return resp->retcode;
498}
499
500static protocol_cmd_t protocol_cmds[] =
501{
502 {SP_CMD_TRANSDTL_ACCOUNT, sp_confirm_transdtl_account},
503 {SP_CMD_HEARTBEAT, sp_confirm_heartbeat},
504 {SP_CMD_LOGIN, sp_confirm_login},
505 {0, NULL}
506};
507
508static uint16 sp_async_confirm_process(sp_pos_t* pos)
509{
510 uint8 ret = 0;
511 uint8 i = 0;
512 int32 timeout_ms = COMM_WAIT_TIME;
513 sp_protocol_response_t resp;
514
515 ret = sp_usart_recv(pos, &resp, timeout_ms);
516 if(ret)
517 {
518 return ret;
519 }
520 while(protocol_cmds[i].func != NULL && protocol_cmds[i].cmd != 0)
521 {
522 if(protocol_cmds[i].cmd == resp.excmd)
523 {
524 return protocol_cmds[i].func(&resp, pos);
525 }
526 ++i;
527 }
528 return 1;
529}
530
531//¼ì²âÉ豸ͨѶ״̬£¬¿ÕÏÐʱ¼ä½øÐÐÁ÷Ë®´¦ÀíµÈ¹¤×÷
532static void sp_check_and_switch_linkstat(sp_pos_t* pos)
533{
534 if(pos->heartbeat.heart_status == HEART_SEND)
535 {
536 //ÒÑ·¢ËÍÐÄÌø£¬Î´ÊÕµ½Ó¦´ð
537 pos->link_stat = 0;
538 pos->heartbeat.heart_status = HEART_INIT;
539 }
540 else if(pos->heartbeat.heart_status == HEART_RECV)
541 {
542 //ÒÑ·¢ËÍÐÄÌø£¬ÊÕµ½Ó¦´ð
543 pos->link_stat = 1;
544 pos->heartbeat.heart_status = HEART_INIT;
545 }
546}
547
548void sp_communicate(sp_pos_t* pos)
549{
550 uint16 ret = 0;
551 uint32 ticker = 0;
552 sp_transdtl_t transdtl;
553
554 if(pos->deviceno == 0)
555 {
556 return;
557 }
558 ticker = sp_get_ticker();
559 if(ticker - pos->last_comm_status.sendtime > COMM_WAIT_TIME)
560 {
561 if(pos->devlogin.last_login_ticker == 0 || pos->devlogin.login_flag == 0
562 || (ticker - pos->devlogin.last_login_ticker) > DELAY_TIME60s*60*24)
563 {
564 pos->devlogin.last_login_ticker = ticker;
565 sp_async_equipment_login(pos);
566 }
567 else if(ticker < pos->heartbeat.last_heartbeat_ticker
568 || pos->heartbeat.last_heartbeat_ticker == 0
569 || (ticker - pos->heartbeat.last_heartbeat_ticker) > DELAY_TIME60s*2)
570 {
571 pos->heartbeat.last_heartbeat_ticker = ticker;
572 sp_async_heartbeat(pos);
573 }
574 else
575 {
576 sp_check_and_switch_linkstat(pos);
577 if(pos->link_stat)
578 {
579 //¼ì²âµ±Ç°ÊÇ·ñÓÐδÉÏ´«µÄÁ÷Ë®
580 if(pos->unconfirm_transdtl.transaddr <= pos->last_transdtl.transaddr)
581 {
582 ret = sp_flash_read((uint32)pos->unconfirm_transdtl.transaddr, (uint8*)&transdtl,
583 sizeof(transdtl));
584 if(!ret)
585 {
586 if(!pos->sysconf.work_mode)
587 {
588 pos->heartbeat.last_heartbeat_ticker = ticker;
589 sp_async_upload_transdtl(pos, &transdtl);
590 }
591 }
592 }
593 }
594 }
595 }
596
597 sp_async_confirm_process(pos);
598}